From 3b04138cdb22211796870ef49d8b3773465641e9 Mon Sep 17 00:00:00 2001 From: Jonathan Gillespie Date: Mon, 2 May 2022 15:01:44 -0400 Subject: [PATCH] Plugin framework v2 + new Big Object Archive plugin + new Log Retention Rules plugin (#298) - Replaced `LoggerSObjectHandlerPlugin` abstract class with new class `LoggerPlugin` that contains 2 interfaces + helper methods, switched to using multiple fields on `LoggerPlugin__mdt` to indicate Apex classes & Flows to run for a plugin, removed SObject-specific fields on `LoggerPlugin__mdt`, added new fields `Log__c.LogPurgeAction__c` and `LoggerSettings__c.DefaultLogPurgeAction__c` - Added support for plugins within LogBatchPurger + added archiving in BigObject plugin (#288) - Added SObject-specific 'execution order' fields on `LoggerPlugin__mdt` - Added picklist values for API versions in calendar year 2022 - Reintroduced the ability to disable trigger handler classes by adding new `LoggerParameter__mdt` records to control via configuration - Renamed CMDT records that control each trigger handler - Added new abstract methods in LoggerSObjectHandler for retrieving each handler's related records in `LoggerParameter__mdt` & `LoggerPlugin__mdt` - Added the ability to configure the list of save methods shown in `loggerSettings` LWC via `LoggerParameter__mdt`, using the prefix 'CustomSaveMethod' - Standardized test-visible method naming conventions to start with `setMock` & updated approaches to use a Map instead of List for CMDT records - Fixed issues in LoggerSObjectHandler for invalid Apex & Flow plugins, started new test class LoggerSObjectHandler_Tests - Added `Log__c` list view to show logs that will be purged in the next 10 days, fixed LogEntryArchiveBuilder using the wrong value for LoggedById**c, updated index on LogEntryArchive**b again (still a WIP), retrieved & formatted metadata for LogEntryArchive\*\*b, updated Admin.profile - Closed #226 by adding new "Log Retentions Rule" plugin, based on my other project, ApexValidationRules - @jamessimone closed #117, and I closed #228 by adding new "Big Object Archiving" plugin - Lowercased the plugin directories - Added plugins folder to the script for generating apex doc files, generated updated docs for Apex - Bumped package version number for the managed package - Added 5 minute wait between unlocked package version creation & package installation, added deprecated `LoggerPlugin__mdt` validation rule & field back into the managed-package folder, cleaned up some docs markdown files - Added tests in LoggerSettingsController_Tests & LogHandler_Tests for new custom setting LoggerSettings**c.DefaultLogPurgeAction**c - Added 'deprecated' to the label of several deprecated fields on `LoggerPlugin__mdt` & removed the related handler methods - More improvements for CMDT records used in the classes LoggerParameter & LoggerPlugin - Added LoggerParameter.matchOnPrefix() to return CMDT records with a specified prefix in the DeveloperName field - Added inner Comparable class in LoggerPlugin to handle custom sorting, since there are some limitations with SOQL queries for CMDT - Classes like LogBatchPurger, LoggerSObjectHandler, and LoggerSettingsController no longer need to track their own mock CMDT records, the LoggerPlugin & LoggerParameter classes now fully handle mocks for their corresponding CMDT objects - LoggerParameter & LoggerPlugin now require mock CMDT records to have DeveloperName populated - they'll throw errors if the field is null - Also added System namespace to some calls for Test.isRunningTest() to handle crazy orgs that have a custom Test class deployed - Added the ability to disable all triggers during tests via new @TestVisible method LoggerSObjectHandler.shouldExecute(Boolean) - Moved LoggerTestUtils to the configuration folder, added public methods for mocking all CMDT records - Consolidated plugin-framework folders back into the configuration folders - Moved demo experience cloud metadata under config, instead of under nebula-logger, updated build.yml to deploy instead of push metadata - Updated test classes to use new overloads for LoggerTestUtils.setMock() - Added new class LoggerDataStore to centralize all database, event bus & queueable operations - Closed #193 by adding classes `LoggerMockDataCreator` for generic mock & data creation (part of configuration layer), `LoggerTestConfigurator` forNebula Logger-specific mocking & setup (part of the log-management layer), and `LoggerMockDataStore` for mocking database & event bus operations throughout the codebase - Started work on #292 by adding businessStatus, complianceGroup & securityClassification nodes to several custom fields (still a WIP) - Rewrote tests in `LogEntryEventBuilder_Tests` to use new methods in `LoggerMockDataCreator` for creating mock instances of database result classes - Scope creep: added fallback fields for setting `RecordName__c` in LogEntryHandler on objects that don't have a display name field, also added some caching - Moved permissionSets and customPermissions folders from configuration to log-management - This should make the configuration folder fully standalone (i.e., it could be deployed by itself) - The perm sets were already referencing all of the objects within log-management, so it seems like it's better to keep the perm sets organized with - Changed labels to just "Entry #" on both `LogEntryEvent__e` and `LogEntryArchive__b` - Fixed an issue in `LogEntryEventBuilder` where some stack trace lines would be duplicated - Scope creeped a bit by adding new fields `DatabaseResultCollectionSize__c` and `RecordCollectionSize__c` - I had planned to add these as part of #222, but decided to add them now so I could duplicate the fields to the `LogEntryArchive__b` big object as well. This also helped to identify (and fix) some issues with missing asserts in LogEntryEventHandler_Tests. - Replaced settings field `IsPlatformEventStorageEnabled__c` with new field `DefaultPlatformEventStorageLocation__**__c`, providing a way for plugins (big object plugin in particular) to provide additional storage location. The big object plugin provides a custom option BIG_OBJECT, in addition to the default value CUSTOM_OBJECTS that will now be used out of the box - Added stack trace fields to the Slack message created via SlackLoggerPlugin - Updated most test classes to use `@IsTest(isParallel=true)`. A few tests (especially integration tests) that do DML on User can't use parallel testing, but most test classes are now using it - Partially implemented #240 by adding HTTP Request & HTTP Response fields to `LogEntryEvent__e`, `LogEntry__c` and `LogEntryArchive__b`. More scope creep yet again, but this will save some headaches long term - since Big Objects can't (easily) have new fields added, I wanted to go ahead and get these new fields added to the data model before the first release of the Big Object plugin. I'll probably add some logging overloads in Logger for these items in a future release, but for now, only LogEntryEventBuilder has new methods for HttpRequest & HttpResponse - Updated some internal debug messages in various places to use Logger's logging methods, instead of `System.debug()` - Deprecated the fields `LogEntryDataMaskRule__mdt.ApplyToMessage__c` and `LogEntryDataMaskRule__mdt.ApplyToRecordJson__c` - Fixed test failures that occur when email deliverability is disabled in an org Co-authored-by: James Simone <16430727+jamessimone@users.noreply.github.com> --- .forceignore | 2 +- .github/codecov.yml | 2 +- .github/workflows/build.yml | 8 +- README.md | 10 +- .../Logger_Test_Site1.site-meta.xml | 0 .../brandingSets/buildYourOwnLWR.json | 0 .../Logger_Test_Site1/config/languages.json | 0 .../config/loggerTestSite.json | 0 .../Logger_Test_Site1/config/mainAppPage.json | 0 .../routes/checkPassword.json | 0 .../Logger_Test_Site1/routes/error.json | 0 .../routes/forgotPassword.json | 0 .../Logger_Test_Site1/routes/home.json | 0 .../Logger_Test_Site1/routes/login.json | 0 .../Logger_Test_Site1/routes/register.json | 0 .../routes/serviceNotAvailable.json | 0 .../themes/buildYourOwnLWR.json | 0 .../views/checkPassword.json | 0 .../Logger_Test_Site1/views/error.json | 0 .../views/forgotPassword.json | 0 .../Logger_Test_Site1/views/home.json | 0 .../Logger_Test_Site1/views/login.json | 0 .../Logger_Test_Site1/views/register.json | 0 .../views/serviceNotAvailable.json | 0 .../Logger Test Site.network-meta.xml | 0 .../pages/BandwidthExceeded.page | 0 .../pages/BandwidthExceeded.page-meta.xml | 0 .../pages/ChangePassword.page | 0 .../pages/ChangePassword.page-meta.xml | 0 .../pages/CommunitiesLanding.page | 0 .../pages/CommunitiesLanding.page-meta.xml | 0 .../pages/CommunitiesLogin.page | 0 .../pages/CommunitiesLogin.page-meta.xml | 0 .../pages/CommunitiesSelfReg.page | 0 .../pages/CommunitiesSelfReg.page-meta.xml | 0 .../pages/CommunitiesSelfRegConfirm.page | 0 .../CommunitiesSelfRegConfirm.page-meta.xml | 0 .../experience-cloud/pages/Exception.page | 0 .../pages/Exception.page-meta.xml | 0 .../experience-cloud/pages/FileNotFound.page | 0 .../pages/FileNotFound.page-meta.xml | 0 .../pages/ForgotPasswordConfirm.page | 0 .../pages/ForgotPasswordConfirm.page-meta.xml | 0 .../experience-cloud/pages/InMaintenance.page | 0 .../pages/InMaintenance.page-meta.xml | 0 .../pages/SiteRegisterConfirm.page | 0 .../pages/SiteRegisterConfirm.page-meta.xml | 0 .../Logger Test Site Profile.profile-meta.xml | 0 .../sites/Logger_Test_Site.site-meta.xml | 0 config/scratch-orgs/base-scratch-def.json | 3 + .../experience-cloud-scratch-def.json | 3 + .../LoggerParameter.md | 20 + docs/apex/Configuration/LoggerPlugin.md | 116 + .../LogBatchPurgeScheduler.md | 0 .../LogBatchPurger.md | 24 +- .../Log-Management/LogEntryEventHandler.md | 45 + .../LogEntryFieldSetPicklist.md | 0 .../LogEntryHandler.md | 0 .../LogEntryTagHandler.md | 0 .../LogHandler.md | 0 .../LogMassDeleteExtension.md | 0 .../Log-Management/LoggerBatchableContext.md | 40 + docs/apex/Log-Management/LoggerEmailSender.md | 35 + .../LoggerEmailUtils.md | 0 .../Log-Management/LoggerSObjectHandler.md | 51 + .../Log-Management/LoggerSObjectMetadata.md | 97 + .../LoggerSettingsController.md | 4 + .../LoggerTagHandler.md | 0 .../LoggerTriggerableContext.md | 57 + .../RelatedLogEntriesController.md | 0 .../ComponentLogger.md | 0 .../FlowCollectionLogEntry.md | 0 .../FlowLogEntry.md | 0 .../FlowLogger.md | 0 .../FlowRecordLogEntry.md | 0 .../LogEntryEventBuilder.md | 40 + .../LogMessage.md | 0 .../Logger.md | 46 +- docs/apex/Logger-Engine/LoggerDataStore.md | 537 + .../Logger-Engine/LoggerSObjectHandler.md | 98 + .../Logger-Engine/LoggerTriggerableContext.md | 59 + docs/apex/Plugins/LogEntryArchiveBuilder.md | 57 + .../apex/Plugins/LogEntryArchiveController.md | 33 + docs/apex/Plugins/LogEntryArchivePlugin.md | 68 + docs/apex/Plugins/LogRetentionRulesPlugin.md | 54 + docs/apex/Plugins/SlackLoggerPlugin.md | 82 + .../Test-Utilities/LoggerMockDataCreator.md | 683 + .../Test-Utilities/LoggerMockDataStore.md | 75 + .../Test-Utilities/LoggerTestConfigurator.md | 163 + docs/apex/index.md | 84 +- .../log-management/LogEntryEventHandler.md | 59 - .../log-management/LoggerSObjectHandler.md | 37 - .../log-management/LoggerSObjectMetadata.md | 85 - .../LoggerSObjectHandlerPlugin.md | 33 - docs/lightning-components/LogEntryBuilder.md | 12 +- docs/lightning-components/Logger.md | 24 +- .../configuration/classes/LoggerParameter.cls | 75 +- .../configuration/classes/LoggerPlugin.cls | 139 + ...cls-meta.xml => LoggerPlugin.cls-meta.xml} | 0 ....IsLogEntryEventHandlerEnabled.md-meta.xml | 17 + ...meter.IsLogEntryHandlerEnabled.md-meta.xml | 17 + ...er.IsLogEntryTagHandlerEnabled.md-meta.xml | 17 + ...rParameter.IsLogHandlerEnabled.md-meta.xml | 17 + ...eter.IsLoggerTagHandlerEnabled.md-meta.xml | 17 + ...er.SendErrorEmailNotifications.md-meta.xml | 2 +- ...ctHandler.LogEntryEventHandler.md-meta.xml | 8 + ...SObjectHandler.LogEntryHandler.md-meta.xml | 8 + ...jectHandler.LogEntryTagHandler.md-meta.xml | 8 + ...oggerSObjectHandler.LogHandler.md-meta.xml | 8 + ...ObjectHandler.LoggerTagHandler.md-meta.xml | 4 + ...ntry Data Mask Rule Layout.layout-meta.xml | 8 - ...__mdt-Logger Plugin Layout.layout-meta.xml | 152 + ...ger SObject Handler Layout.layout-meta.xml | 24 +- .../fields/ApplyToMessage__c.field-meta.xml | 3 +- .../ApplyToRecordJson__c.field-meta.xml | 1 + .../listViews/All.listView-meta.xml | 2 - .../fields/Description__c.field-meta.xml | 2 + .../fields/Value__c.field-meta.xml | 2 + .../listViews/All.listView-meta.xml | 1 + .../LoggerPlugin__mdt.object-meta.xml | 0 .../BatchPurgerApexClass__c.field-meta.xml | 11 + ...tchPurgerExecutionOrder__c.field-meta.xml} | 4 +- .../BatchPurgerFlowName__c.field-meta.xml | 11 + .../fields/Description__c.field-meta.xml | 0 .../fields/ExecutionOrder__c.field-meta.xml | 14 + .../fields/IsEnabled__c.field-meta.xml | 0 .../fields/Link__c.field-meta.xml | 10 + .../fields/PluginApiName__c.field-meta.xml | 13 + .../fields/PluginType__c.field-meta.xml | 9 +- .../SObjectHandlerApexClass__c.field-meta.xml | 11 + ...ectHandlerExecutionOrder__c.field-meta.xml | 16 + .../SObjectHandlerFlowName__c.field-meta.xml | 11 + .../fields/SObjectType__c.field-meta.xml | 8 +- .../fields/VersionNumber__c.field-meta.xml | 12 + .../listViews/All.listView-meta.xml | 13 + ..._are_not_supported.validationRule-meta.xml | 9 + .../LoggerSObjectHandler__mdt.object-meta.xml | 4 +- .../fields/IsEnabled__c.field-meta.xml | 0 .../SObjectHandlerApexClass__c.field-meta.xml | 13 + .../fields/SObjectType__c.field-meta.xml | 0 .../listViews/All.listView-meta.xml | 1 + .../DefaultLogPurgeAction__c.field-meta.xml | 14 + ...faultLogShareAccessLevel__c.field-meta.xml | 3 +- ...formEventStorageLocation__c.field-meta.xml | 16 + .../fields/IsSavingEnabled__c.field-meta.xml | 1 + .../applications/LoggerConsole.app-meta.xml | 1 + .../classes/LogBatchPurgeScheduler.cls | 2 +- .../log-management/classes/LogBatchPurger.cls | 167 +- .../classes/LogEntryEventHandler.cls | 86 +- .../classes/LogEntryHandler.cls | 38 +- .../log-management/classes/LogHandler.cls | 39 +- .../classes/LogMassDeleteExtension.cls | 2 +- .../classes/LoggerBatchableContext.cls | 30 + ...ml => LoggerBatchableContext.cls-meta.xml} | 0 .../classes/LoggerEmailSender.cls} | 51 +- .../classes/LoggerEmailSender.cls-meta.xml} | 0 .../classes/LoggerSObjectHandler.cls | 182 - .../classes/LoggerSObjectMetadata.cls | 32 +- .../classes/LoggerSettingsController.cls | 72 +- ...fyLoggerSettings.customPermission-meta.xml | 0 .../LogEntryRecordPage.flexipage-meta.xml | 196 + .../LogRecordPage.flexipage-meta.xml | 10 + ...gEntry__c-Log Entry Layout.layout-meta.xml | 60 +- .../layouts/Log__c-Log Layout.layout-meta.xml | 6 +- ...EventSchema.json => getSchemaForName.json} | 0 .../__tests__/logEntryEventStream.test.js | 26 +- .../logEntryEventStream.js | 4 +- .../__tests__/data/getObjectInfo.json | 822 - .../__tests__/data/getPicklistOptions.json | 8 + ...tingsSchema.json => getSchemaForName.json} | 14 +- .../__tests__/loggerSettings.test.js | 10 +- .../lwc/loggerSettings/loggerSettings.js | 22 +- .../loggerSettingsPageLayout.js | 8 +- .../__tests__/relatedLogEntries.test.js | 1 + .../fields/LogEntryOrigin__c.field-meta.xml | 2 + .../LogEntryTimestamp__c.field-meta.xml | 2 + .../fields/LogEntry__c.field-meta.xml | 2 + .../fields/LogLink__c.field-meta.xml | 2 + .../LoggedByUsernameLink__c.field-meta.xml | 3 + .../fields/Tag__c.field-meta.xml | 2 + .../fields/UniqueId__c.field-meta.xml | 2 + ...baseResultCollectionSize__c.field-meta.xml | 15 + .../fields/EventUuid__c.field-meta.xml | 2 +- .../HttpRequestBodyMasked__c.field-meta.xml | 11 + .../fields/HttpRequestBody__c.field-meta.xml | 11 + .../HttpRequestCompressed__c.field-meta.xml | 10 + .../HttpRequestEndpoint__c.field-meta.xml | 12 + .../HttpRequestMethod__c.field-meta.xml | 51 + .../HttpResponseBodyMasked__c.field-meta.xml | 11 + .../fields/HttpResponseBody__c.field-meta.xml | 11 + .../HttpResponseHeaderKeys__c.field-meta.xml | 11 + .../HttpResponseStatusCode__c.field-meta.xml | 13 + .../HttpResponseStatus__c.field-meta.xml | 12 + .../RecordCollectionSize__c.field-meta.xml | 15 + .../fields/ApiReleaseNumber__c.field-meta.xml | 2 + .../ApiReleaseVersion__c.field-meta.xml | 2 + .../fields/ApiVersion__c.field-meta.xml | 2 + .../Log__c/fields/ClosedBy__c.field-meta.xml | 2 + .../fields/ClosedDate__c.field-meta.xml | 2 + .../Log__c/fields/Comments__c.field-meta.xml | 2 + .../Log__c/fields/EndTime__c.field-meta.xml | 2 + .../Log__c/fields/IsClosed__c.field-meta.xml | 2 + .../fields/IsResolved__c.field-meta.xml | 2 + .../Log__c/fields/Issue__c.field-meta.xml | 4 +- .../Log__c/fields/Locale__c.field-meta.xml | 2 + .../LogEntriesSummary__c.field-meta.xml | 2 + .../fields/LogPurgeAction__c.field-meta.xml | 25 + .../fields/LogRetentionDate__c.field-meta.xml | 2 + .../LoggedByUsernameLink__c.field-meta.xml | 3 + .../fields/LoggedByUsername__c.field-meta.xml | 3 + .../Log__c/fields/LoggedBy__c.field-meta.xml | 2 + .../LoggerVersionNumber__c.field-meta.xml | 3 +- .../fields/LoginApplication__c.field-meta.xml | 2 + .../fields/LoginBrowser__c.field-meta.xml | 2 + .../fields/LoginDomain__c.field-meta.xml | 2 + .../fields/LoginHistoryId__c.field-meta.xml | 2 + .../fields/LoginPlatform__c.field-meta.xml | 2 + .../Log__c/fields/LoginType__c.field-meta.xml | 4 +- .../Log__c/fields/LogoutUrl__c.field-meta.xml | 2 + ...EntryLoggingLevelOrdinal__c.field-meta.xml | 2 + .../Log__c/fields/NetworkId__c.field-meta.xml | 3 + .../fields/NetworkLoginUrl__c.field-meta.xml | 3 + .../fields/NetworkLogoutUrl__c.field-meta.xml | 3 + .../fields/NetworkName__c.field-meta.xml | 3 + ...tworkSelfRegistrationUrl__c.field-meta.xml | 3 + .../NetworkUrlPathPrefix__c.field-meta.xml | 3 + .../OrganizationDomainUrl__c.field-meta.xml | 2 + ...anizationEnvironmentType__c.field-meta.xml | 2 + .../fields/OrganizationId__c.field-meta.xml | 2 + ...OrganizationInstanceName__c.field-meta.xml | 2 + ...tionInstanceReleaseCycle__c.field-meta.xml | 2 + .../fields/OrganizationName__c.field-meta.xml | 2 + ...anizationNamespacePrefix__c.field-meta.xml | 2 + .../fields/OrganizationType__c.field-meta.xml | 2 + .../Log__c/fields/ParentLog__c.field-meta.xml | 2 + .../Log__c/fields/Priority__c.field-meta.xml | 2 + .../Log__c/fields/ProfileId__c.field-meta.xml | 3 + .../fields/ProfileLink__c.field-meta.xml | 3 + .../fields/ProfileName__c.field-meta.xml | 3 + .../Log__c/fields/Scenario__c.field-meta.xml | 3 +- .../Log__c/fields/SessionId__c.field-meta.xml | 3 + .../SessionSecurityLevel__c.field-meta.xml | 3 + .../fields/SessionType__c.field-meta.xml | 5 +- .../Log__c/fields/SourceIp__c.field-meta.xml | 3 + .../Log__c/fields/StartTime__c.field-meta.xml | 2 + .../Log__c/fields/Status__c.field-meta.xml | 2 + .../SystemModeSummary__c.field-meta.xml | 2 + .../fields/SystemMode__c.field-meta.xml | 4 +- .../fields/ThemeDisplayed__c.field-meta.xml | 2 + .../fields/TimeZoneId__c.field-meta.xml | 3 + .../fields/TimeZoneName__c.field-meta.xml | 2 + .../TotalDEBUGLogEntries__c.field-meta.xml | 2 + .../TotalERRORLogEntries__c.field-meta.xml | 2 + .../TotalFINELogEntries__c.field-meta.xml | 2 + .../TotalFINERLogEntries__c.field-meta.xml | 2 + .../TotalFINESTLogEntries__c.field-meta.xml | 2 + .../TotalINFOLogEntries__c.field-meta.xml | 2 + .../TotalLimitsCpuTimeUsed__c.field-meta.xml | 2 + .../fields/TotalLogEntries__c.field-meta.xml | 2 + .../TotalWARNLogEntries__c.field-meta.xml | 2 + .../fields/TransactionId__c.field-meta.xml | 2 + ...UserLicenseDefinitionKey__c.field-meta.xml | 3 + .../fields/UserLicenseId__c.field-meta.xml | 3 + .../fields/UserLicenseName__c.field-meta.xml | 5 +- .../UserLoggingLevelOrdinal__c.field-meta.xml | 3 + .../fields/UserLoggingLevel__c.field-meta.xml | 3 + .../fields/UserRoleId__c.field-meta.xml | 3 + .../fields/UserRoleLink__c.field-meta.xml | 3 + .../fields/UserRoleName__c.field-meta.xml | 3 + .../Log__c/fields/UserType__c.field-meta.xml | 3 + .../WasLoggedByCurrentUser__c.field-meta.xml | 3 + .../LogsToPurgeSoon.listView-meta.xml | 25 + .../fields/TotalLogEntries__c.field-meta.xml | 2 + .../fields/UniqueId__c.field-meta.xml | 2 + .../LoggerAdmin.permissionset-meta.xml | 60 + .../LoggerEndUser.permissionset-meta.xml | 19 +- .../LoggerLogCreator.permissionset-meta.xml | 0 .../LoggerLogViewer.permissionset-meta.xml | 64 +- .../Log__c.Manage.quickAction-meta.xml | 5 + .../main/log-management/triggers/Log.trigger | 2 +- .../log-management/triggers/LogEntry.trigger | 2 +- .../triggers/LogEntryEvent.trigger | 2 +- .../triggers/LogEntryTag.trigger | 2 +- .../log-management/triggers/LoggerTag.trigger | 2 +- .../logger-engine/classes/ComponentLogger.cls | 5 +- .../classes/FlowCollectionLogEntry.cls | 7 +- .../classes/LogEntryEventBuilder.cls | 134 +- .../main/logger-engine/classes/Logger.cls | 174 +- .../logger-engine/classes/LoggerDataStore.cls | 295 + .../classes/LoggerDataStore.cls-meta.xml} | 0 .../classes/LoggerSObjectHandler.cls | 287 + .../LoggerSObjectHandler.cls-meta.xml} | 0 .../classes/LoggerTriggerableContext.cls | 87 + .../LoggerTriggerableContext.cls-meta.xml} | 0 ...baseResultCollectionSize__c.field-meta.xml | 15 + .../HttpRequestBodyMasked__c.field-meta.xml | 11 + .../fields/HttpRequestBody__c.field-meta.xml | 11 + .../HttpRequestCompressed__c.field-meta.xml | 10 + .../HttpRequestEndpoint__c.field-meta.xml | 12 + .../HttpRequestMethod__c.field-meta.xml | 12 + .../HttpResponseBodyMasked__c.field-meta.xml | 11 + .../fields/HttpResponseBody__c.field-meta.xml | 11 + .../HttpResponseHeaderKeys__c.field-meta.xml | 11 + .../HttpResponseStatusCode__c.field-meta.xml | 13 + .../HttpResponseStatus__c.field-meta.xml | 12 + .../fields/LoginDomain__c.field-meta.xml | 1 + .../RecordCollectionSize__c.field-meta.xml | 15 + .../fields/Topics__c.field-meta.xml | 1 + .../TransactionEntryNumber__c.field-meta.xml | 2 +- .../classes/LoggerSObjectHandlerPlugin.cls | 41 - .../fields/PluginApiName__c.field-meta.xml | 27 - ..._are_not_supported.validationRule-meta.xml | 14 - .../core/tests/LoggerCore.testSuite-meta.xml | 30 + .../tests/common/classes/LoggerTestUtils.cls | 256 - .../classes/LoggerEmailUtils_Tests.cls | 102 - .../classes/LoggerParameter_Tests.cls | 191 +- .../classes/LoggerPlugin_Tests.cls | 123 + .../classes/LoggerPlugin_Tests.cls-meta.xml} | 0 .../LoggerConfiguration.testSuite-meta.xml | 1 + .../utilities/LoggerMockDataCreator.cls | 678 + .../LoggerMockDataCreator.cls-meta.xml} | 0 .../classes/LogBatchPurgeScheduler_Tests.cls | 7 +- .../classes/LogBatchPurger_Tests.cls | 305 +- .../classes/LogEntryEventHandler_Tests.cls | 1010 +- .../LogEntryFieldSetPicklist_Tests.cls | 2 +- .../classes/LogEntryHandler_Tests.cls | 225 +- .../classes/LogEntryTagHandler_Tests.cls | 73 +- .../classes/LogHandler_Tests.cls | 334 +- .../classes/LogMassDeleteExtension_Tests.cls | 20 +- .../classes/LoggerBatchableContext_Tests.cls | 32 + ...LoggerBatchableContext_Tests.cls-meta.xml} | 0 .../classes/LoggerEmailSender_Tests.cls | 134 + .../LoggerEmailSender_Tests.cls-meta.xml | 5 + .../classes/LoggerSObjectMetadata_Tests.cls | 23 +- .../LoggerSettingsController_Tests.cls | 92 +- .../classes/LoggerTagHandler_Tests.cls | 72 +- .../RelatedLogEntriesController_Tests.cls | 8 +- .../LoggerLogManagement.testSuite-meta.xml | 2 + .../utilities/LoggerTestConfigurator.cls | 162 + .../LoggerTestConfigurator.cls-meta.xml | 5 + .../classes/ComponentLogger_Tests.cls | 282 +- .../classes/FlowCollectionLogEntry_Tests.cls | 319 +- .../classes/FlowLogEntry_Tests.cls | 248 +- .../classes/FlowLogger_Tests.cls | 126 +- .../classes/FlowRecordLogEntry_Tests.cls | 277 +- .../classes/LogEntryEventBuilder_Tests.cls | 637 +- .../classes/LogMessage_Tests.cls | 25 +- .../classes/LoggerDataStore_Tests.cls | 608 + .../LoggerDataStore_Tests.cls-meta.xml | 5 + .../classes/LoggerSObjectHandler_Tests.cls | 416 + .../LoggerSObjectHandler_Tests.cls-meta.xml | 5 + .../LoggerTriggerableContext_Tests.cls | 97 + ...oggerTriggerableContext_Tests.cls-meta.xml | 5 + .../logger-engine/classes/Logger_Tests.cls | 768 +- .../LoggerEngine.testSuite-meta.xml | 3 + .../utilities/LoggerMockDataStore.cls | 220 + .../LoggerMockDataStore.cls-meta.xml | 5 + .../LoggerSObjectHandlerPlugin_Tests.cls | 43 - .../LoggerPluginFramework.testSuite-meta.xml | 5 - .../flows/LogHandler_Tests_Flow.flow-meta.xml | 213 - .../MockLogBatchPurgerPlugin.flow-meta.xml | 70 + ...ckLoggerSObjectHandlerPlugin.flow-meta.xml | 70 + .../tests/LogBatchPurger_Tests_Flow.cls | 208 + .../LogBatchPurger_Tests_Flow.cls-meta.xml | 5 + .../LogBatchPurger_Tests_Integration.cls | 2 +- ...LogEntryEventBuilder_Tests_Integration.cls | 28 +- .../tests/LogEntryHandler_Tests_Flow.cls | 11 +- .../LogEntryHandler_Tests_Integration.cls | 33 + ...ntryHandler_Tests_Integration.cls-meta.xml | 5 + .../tests/LogHandler_Tests_Flow.cls | 33 - ...gMassDeleteExtension_Tests_Integration.cls | 12 +- .../tests/LoggerSObjectHandler_Tests_Flow.cls | 43 + ...ggerSObjectHandler_Tests_Flow.cls-meta.xml | 5 + ...oggerSettingsController_Tests_Security.cls | 22 +- .../tests/Logger_Tests_ExperienceSite.cls | 6 +- .../Logger_Tests_InboundEmailHandler.cls | 26 +- .../tests/Logger_Tests_MergeResult.cls | 44 +- .../EnableStatusApiCallout__c.field-meta.xml | 1 + .../EnableSystemMessages__c.field-meta.xml | 1 + ...tformEventStorageEnabled__c.field-meta.xml | 4 +- .../deprecated/classes/LoggerEmailUtils.cls | 3 + .../classes/LoggerEmailUtils.cls-meta.xml | 5 + .../classes/LoggerEmailUtils_Tests.cls | 4 + .../LoggerEmailUtils_Tests.cls-meta.xml | 5 + .../classes/LoggerSObjectHandlerPlugin.cls | 3 + .../LoggerSObjectHandlerPlugin.cls-meta.xml | 5 + .../LoggerSObjectHandlerPlugin_Tests.cls | 4 + ...gerSObjectHandlerPlugin_Tests.cls-meta.xml | 5 + .../deprecated/classes/LoggerTestUtils.cls | 4 + .../classes/LoggerTestUtils.cls-meta.xml | 5 + .../managed-package/sfdx-project.json | 6 +- ...nstall-unlocked-package-plugin-sandbox.png | Bin 0 -> 39945 bytes .../btn-install-unlocked-package-plugin.png | Bin .../slack/classes/SlackLoggerPlugin_Tests.cls | 186 - .../LoggerPlugin.Slack.md-meta.xml | 27 - .../.images/log-entry-archives-tab.png | Bin 0 -> 226207 bytes .../plugins/big-object-archiving/README.md | 32 + .../plugin/classes/LogEntryArchiveBuilder.cls | 325 + .../LogEntryArchiveBuilder.cls-meta.xml | 5 + .../classes/LogEntryArchiveBuilder_Tests.cls | 863 + .../LogEntryArchiveBuilder_Tests.cls-meta.xml | 5 + .../classes/LogEntryArchiveController.cls | 102 + .../LogEntryArchiveController.cls-meta.xml | 5 + .../LogEntryArchiveController_Tests.cls | 98 + ...gEntryArchiveController_Tests.cls-meta.xml | 5 + .../plugin/classes/LogEntryArchivePlugin.cls | 231 + .../LogEntryArchivePlugin.cls-meta.xml | 5 + .../classes/LogEntryArchivePlugin_Tests.cls | 239 + .../LogEntryArchivePlugin_Tests.cls-meta.xml | 5 + ...er.CustomLogPurgeActionArchive.md-meta.xml | 17 + ...omSaveMethodBigObjectImmediate.md-meta.xml | 17 + ...omSaveMethodBigObjectQueueable.md-meta.xml | 17 + ...CustomStorageLocationBigObject.md-meta.xml | 19 + .../LoggerPlugin.LogEntryArchive.md-meta.xml | 49 + ...oggerSObjectMetadata.getSchemaForName.json | 799 + .../__tests__/logEntryArchives.test.js | 261 + .../logEntryArchives/logEntryArchives.html | 88 + .../lwc/logEntryArchives/logEntryArchives.js | 195 + .../logEntryArchives.js-meta.xml | 9 + .../LogEntryArchive__b.object-meta.xml | 8 + .../fields/ApiReleaseNumber__c.field-meta.xml | 14 + .../ApiReleaseVersion__c.field-meta.xml | 14 + .../fields/ApiVersion__c.field-meta.xml | 10 + .../ArchiveRetentionDate__c.field-meta.xml | 11 + .../fields/ArchivedById__c.field-meta.xml | 10 + .../ArchivedByUsername__c.field-meta.xml | 12 + .../fields/ArchivedDate__c.field-meta.xml | 10 + .../fields/ClosedById__c.field-meta.xml | 10 + .../fields/ClosedByUsername__c.field-meta.xml | 12 + .../fields/ClosedDate__c.field-meta.xml | 10 + .../fields/Comments__c.field-meta.xml | 11 + .../fields/ComponentType__c.field-meta.xml | 10 + ...baseResultCollectionSize__c.field-meta.xml | 13 + ...baseResultCollectionType__c.field-meta.xml | 10 + .../DatabaseResultJson__c.field-meta.xml | 9 + .../DatabaseResultType__c.field-meta.xml | 10 + .../fields/EpochTimestamp__c.field-meta.xml | 12 + .../fields/EventUuid__c.field-meta.xml | 13 + .../fields/ExceptionMessage__c.field-meta.xml | 9 + .../ExceptionStackTrace__c.field-meta.xml | 9 + .../fields/ExceptionType__c.field-meta.xml | 10 + .../HttpRequestBodyMasked__c.field-meta.xml | 12 + .../fields/HttpRequestBody__c.field-meta.xml | 11 + .../HttpRequestCompressed__c.field-meta.xml | 12 + .../HttpRequestEndpoint__c.field-meta.xml | 12 + .../HttpRequestMethod__c.field-meta.xml | 12 + .../HttpResponseBodyMasked__c.field-meta.xml | 12 + .../fields/HttpResponseBody__c.field-meta.xml | 11 + .../HttpResponseHeaderKeys__c.field-meta.xml | 11 + .../HttpResponseStatusCode__c.field-meta.xml | 13 + .../HttpResponseStatus__c.field-meta.xml | 12 + .../fields/IsClosed__c.field-meta.xml | 12 + .../fields/IsResolved__c.field-meta.xml | 12 + .../fields/Issue__c.field-meta.xml | 12 + ...imitsAggregateQueriesMax__c.field-meta.xml | 11 + ...mitsAggregateQueriesUsed__c.field-meta.xml | 11 + .../LimitsAggregateQueryMax__c.field-meta.xml | 11 + .../LimitsAsyncCallsMax__c.field-meta.xml | 11 + .../LimitsAsyncCallsUsed__c.field-meta.xml | 11 + .../LimitsCalloutsMax__c.field-meta.xml | 11 + .../LimitsCalloutsUsed__c.field-meta.xml | 11 + .../fields/LimitsCpuTimeMax__c.field-meta.xml | 11 + .../LimitsCpuTimeUsed__c.field-meta.xml | 11 + .../fields/LimitsDmlRowsMax__c.field-meta.xml | 11 + .../LimitsDmlRowsUsed__c.field-meta.xml | 11 + .../LimitsDmlStatementsMax__c.field-meta.xml | 11 + .../LimitsDmlStatementsUsed__c.field-meta.xml | 11 + ...imitsEmailInvocationsMax__c.field-meta.xml | 11 + ...mitsEmailInvocationsUsed__c.field-meta.xml | 11 + .../LimitsFutureCallsMax__c.field-meta.xml | 11 + .../LimitsFutureCallsUsed__c.field-meta.xml | 11 + .../LimitsHeapSizeMax__c.field-meta.xml | 11 + .../LimitsHeapSizeUsed__c.field-meta.xml | 11 + ...tsMobilePushApexCallsMax__c.field-meta.xml | 11 + ...sMobilePushApexCallsUsed__c.field-meta.xml | 11 + ...mmediateDmlStatementsMax__c.field-meta.xml | 11 + ...mediateDmlStatementsUsed__c.field-meta.xml | 11 + .../LimitsQueueableJobsMax__c.field-meta.xml | 11 + .../LimitsQueueableJobsUsed__c.field-meta.xml | 11 + .../LimitsSoqlQueriesMax__c.field-meta.xml | 11 + .../LimitsSoqlQueriesUsed__c.field-meta.xml | 11 + ...sSoqlQueryLocatorRowsMax__c.field-meta.xml | 11 + ...SoqlQueryLocatorRowsUsed__c.field-meta.xml | 11 + .../LimitsSoqlQueryRowsMax__c.field-meta.xml | 11 + .../LimitsSoqlQueryRowsUsed__c.field-meta.xml | 11 + .../LimitsSoslSearchesMax__c.field-meta.xml | 11 + .../LimitsSoslSearchesUsed__c.field-meta.xml | 11 + .../fields/Locale__c.field-meta.xml | 10 + .../fields/LogEntryName__c.field-meta.xml | 10 + .../fields/LogName__c.field-meta.xml | 10 + .../fields/LogPurgeAction__c.field-meta.xml | 12 + .../fields/LogRetentionDate__c.field-meta.xml | 13 + .../fields/LoggedById__c.field-meta.xml | 10 + .../fields/LoggedByUsername__c.field-meta.xml | 10 + .../LoggerVersionNumber__c.field-meta.xml | 10 + .../LoggingLevelOrdinal__c.field-meta.xml | 11 + .../fields/LoggingLevel__c.field-meta.xml | 10 + .../fields/LoginApplication__c.field-meta.xml | 10 + .../fields/LoginBrowser__c.field-meta.xml | 10 + .../fields/LoginDomain__c.field-meta.xml | 11 + .../fields/LoginHistoryId__c.field-meta.xml | 10 + .../fields/LoginPlatform__c.field-meta.xml | 10 + .../fields/LoginType__c.field-meta.xml | 10 + .../fields/LogoutUrl__c.field-meta.xml | 10 + .../fields/MessageMasked__c.field-meta.xml | 12 + .../fields/MessageTruncated__c.field-meta.xml | 12 + .../fields/Message__c.field-meta.xml | 9 + .../fields/NetworkId__c.field-meta.xml | 10 + .../fields/NetworkLoginUrl__c.field-meta.xml | 10 + .../fields/NetworkLogoutUrl__c.field-meta.xml | 10 + .../fields/NetworkName__c.field-meta.xml | 11 + ...tworkSelfRegistrationUrl__c.field-meta.xml | 10 + .../NetworkUrlPathPrefix__c.field-meta.xml | 12 + .../OrganizationDomainUrl__c.field-meta.xml | 11 + ...anizationEnvironmentType__c.field-meta.xml | 10 + .../fields/OrganizationId__c.field-meta.xml | 10 + ...OrganizationInstanceName__c.field-meta.xml | 10 + ...tionInstanceReleaseCycle__c.field-meta.xml | 12 + .../fields/OrganizationName__c.field-meta.xml | 10 + ...anizationNamespacePrefix__c.field-meta.xml | 10 + .../fields/OrganizationType__c.field-meta.xml | 10 + .../fields/OriginLocation__c.field-meta.xml | 10 + .../fields/OriginType__c.field-meta.xml | 10 + .../ParentLogTransactionId__c.field-meta.xml | 10 + .../fields/Priority__c.field-meta.xml | 12 + .../fields/ProfileId__c.field-meta.xml | 10 + .../fields/ProfileName__c.field-meta.xml | 10 + .../RecordCollectionSize__c.field-meta.xml | 13 + .../RecordCollectionType__c.field-meta.xml | 10 + .../fields/RecordId__c.field-meta.xml | 12 + .../fields/RecordJsonMasked__c.field-meta.xml | 12 + .../fields/RecordJson__c.field-meta.xml | 9 + .../fields/RecordName__c.field-meta.xml | 12 + ...ordSObjectClassification__c.field-meta.xml | 10 + ...cordSObjectTypeNamespace__c.field-meta.xml | 10 + .../RecordSObjectType__c.field-meta.xml | 10 + .../fields/Scenario__c.field-meta.xml | 10 + .../fields/SessionId__c.field-meta.xml | 10 + .../SessionSecurityLevel__c.field-meta.xml | 10 + .../fields/SessionType__c.field-meta.xml | 10 + .../fields/SourceIp__c.field-meta.xml | 10 + .../fields/StackTrace__c.field-meta.xml | 9 + .../fields/Status__c.field-meta.xml | 12 + .../fields/SystemMode__c.field-meta.xml | 10 + .../fields/Tags__c.field-meta.xml | 9 + .../fields/ThemeDisplayed__c.field-meta.xml | 10 + .../fields/TimeZoneId__c.field-meta.xml | 10 + .../fields/TimeZoneName__c.field-meta.xml | 10 + .../fields/TimestampString__c.field-meta.xml | 12 + .../fields/Timestamp__c.field-meta.xml | 8 + .../TransactionEntryNumber__c.field-meta.xml | 12 + .../fields/TransactionId__c.field-meta.xml | 10 + .../TriggerIsExecuting__c.field-meta.xml | 12 + .../TriggerOperationType__c.field-meta.xml | 10 + .../TriggerSObjectType__c.field-meta.xml | 10 + ...UserLicenseDefinitionKey__c.field-meta.xml | 11 + .../fields/UserLicenseId__c.field-meta.xml | 10 + .../fields/UserLicenseName__c.field-meta.xml | 10 + .../UserLoggingLevelOrdinal__c.field-meta.xml | 11 + .../fields/UserLoggingLevel__c.field-meta.xml | 10 + .../fields/UserRoleId__c.field-meta.xml | 10 + .../fields/UserRoleName__c.field-meta.xml | 10 + .../fields/UserType__c.field-meta.xml | 10 + .../LogEntryArchiveIndex.indexe-meta.xml | 17 + ...ogEntryArchiveAdmin.permissionset-meta.xml | 674 + .../plugin/tabs/LogEntryArchives.tab-meta.xml | 6 + ...erLogEntryArchivePlugin.testSuite-meta.xml | 6 + ...ple-log-retention-rule-with-conditions.png | Bin 0 -> 186103 bytes .../example-log-retention-rules-list-view.png | Bin 0 -> 161345 bytes .../plugins/log-retention-rules/README.md | 42 + .../plugin/classes/LogRetentionFilter.cls | 469 + .../classes/LogRetentionFilter.cls-meta.xml | 5 + .../classes/LogRetentionFilter_Tests.cls | 476 + .../LogRetentionFilter_Tests.cls-meta.xml | 5 + .../classes/LogRetentionRulesPlugin.cls | 130 + .../LogRetentionRulesPlugin.cls-meta.xml | 5 + .../classes/LogRetentionRulesPlugin_Tests.cls | 211 + ...LogRetentionRulesPlugin_Tests.cls-meta.xml | 5 + ...nRule.Sample_Rule_1_Error_Logs.md-meta.xml | 29 + ...e_Rule_1_Scenarios_With_Errors.md-meta.xml | 29 + ...e_Rule_2_Additional_Error_Logs.md-meta.xml | 29 + ....Sample_Rule_1_Error_Condition.md-meta.xml | 33 + ...le_Rule_1_Scenario_A_Condition.md-meta.xml | 33 + ...le_Rule_1_Scenario_B_Condition.md-meta.xml | 33 + ...dition.Sample_Rule_1_condition.md-meta.xml | 33 + ....Sample_Rule_2_Error_Condition.md-meta.xml | 33 + ...LoggerPlugin.LogRetentionRules.md-meta.xml | 51 + ...ion Rule Condition Layout.layout-meta.xml} | 43 +- ...-Log Retention Rule Layout.layout-meta.xml | 116 + ...etentionRuleCondition__mdt.object-meta.xml | 8 + .../fields/FieldPath__c.field-meta.xml | 12 + .../fields/LogRetentionRule__c.field-meta.xml | 13 + .../fields/Operator__c.field-meta.xml | 60 + .../fields/SortOrder__c.field-meta.xml | 12 + .../fields/ValueType__c.field-meta.xml | 30 + .../fields/Value__c.field-meta.xml | 11 + .../listViews/All.listView-meta.xml | 14 + .../LogRetentionRule__mdt.object-meta.xml | 8 + .../ConditionLogicType__c.field-meta.xml | 30 + .../CustomConditionLogic__c.field-meta.xml | 11 + .../fields/ExecutionOrder__c.field-meta.xml | 14 + .../fields/IsEnabled__c.field-meta.xml | 9 + ...NumberOfDaysToRetainLogs__c.field-meta.xml | 14 + .../listViews/All.listView-meta.xml | 7 +- ...LogRetentionRulesPlugin.testSuite-meta.xml | 5 + .../btn-install-unlocked-package-plugin.png | Bin .../.images}/logger-admin-dashboard.png | Bin .../README.md | 4 +- .../LogDashboards.dashboardFolder-meta.xml | 0 .../LoggerAdmin.dashboard-meta.xml | 1 + ...EntryDailyRetentionSummary.report-meta.xml | 0 .../LogEntryDailySummary.report-meta.xml | 0 .../LogEntryOriginSummary.report-meta.xml | 0 .../LogEntryScenarioSummary.report-meta.xml | 0 .../LogEntrySummary.report-meta.xml | 0 .../.images}/slack-plugin-notification.png | Bin .../.images}/slack-plugin-parameters.png | Bin .../plugins/{Slack => slack}/README.md | 10 +- .../plugin/core/main/README.md | 0 .../SendSlackNotification__c.field-meta.xml | 0 .../SlackNotificationDate__c.field-meta.xml | 0 .../AllLogsSentToSlack.listView-meta.xml | 0 .../AllLogsToBeSentToSlack.listView-meta.xml | 0 .../slack/classes/SlackLoggerPlugin.cls | 99 +- .../classes/SlackLoggerPlugin.cls-meta.xml | 5 + .../slack/classes/SlackLoggerPlugin_Tests.cls | 229 + .../SlackLoggerPlugin_Tests.cls-meta.xml | 5 + .../LoggerParameter.SlackEndpoint.md-meta.xml | 0 ....SlackNotificationLoggingLevel.md-meta.xml | 0 .../LoggerPlugin.Slack.md-meta.xml | 51 + ...gerSlackPluginAdmin.permissionset-meta.xml | 0 .../Slack.remoteSite-meta.xml | 0 .../classes/ExampleBigObjectDataGenerator.cls | 22 + ...ExampleBigObjectDataGenerator.cls-meta.xml | 5 + .../recipes/profiles/Admin.profile-meta.xml | 1755 +- package-lock.json | 24150 ++++++++++++---- package.json | 20 +- .../create-and-install-package-version.ps1 | 2 +- .../create-managed-package-beta-version.ps1 | 2 +- scripts/build/generate-apex-docs.ps1 | 24 +- scripts/data/create-sample-log-entries.apex | 30 +- .../create-sample-log-entry-archives.apex | 18 + .../query-log-entry-archive-big-object.apex | 15 + sfdx-project.json | 51 +- 644 files changed, 39139 insertions(+), 11449 deletions(-) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1.site-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/brandingSets/buildYourOwnLWR.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/config/languages.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/config/loggerTestSite.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/config/mainAppPage.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/routes/checkPassword.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/routes/error.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/routes/forgotPassword.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/routes/home.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/routes/login.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/routes/register.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/routes/serviceNotAvailable.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/themes/buildYourOwnLWR.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/views/checkPassword.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/views/error.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/views/forgotPassword.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/views/home.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/views/login.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/views/register.json (100%) rename {nebula-logger => config}/experience-cloud/experiences/Logger_Test_Site1/views/serviceNotAvailable.json (100%) rename {nebula-logger => config}/experience-cloud/networks/Logger Test Site.network-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/BandwidthExceeded.page (100%) rename {nebula-logger => config}/experience-cloud/pages/BandwidthExceeded.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/ChangePassword.page (100%) rename {nebula-logger => config}/experience-cloud/pages/ChangePassword.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesLanding.page (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesLanding.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesLogin.page (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesLogin.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesSelfReg.page (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesSelfReg.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesSelfRegConfirm.page (100%) rename {nebula-logger => config}/experience-cloud/pages/CommunitiesSelfRegConfirm.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/Exception.page (100%) rename {nebula-logger => config}/experience-cloud/pages/Exception.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/FileNotFound.page (100%) rename {nebula-logger => config}/experience-cloud/pages/FileNotFound.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/ForgotPasswordConfirm.page (100%) rename {nebula-logger => config}/experience-cloud/pages/ForgotPasswordConfirm.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/InMaintenance.page (100%) rename {nebula-logger => config}/experience-cloud/pages/InMaintenance.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/pages/SiteRegisterConfirm.page (100%) rename {nebula-logger => config}/experience-cloud/pages/SiteRegisterConfirm.page-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/profiles/Logger Test Site Profile.profile-meta.xml (100%) rename {nebula-logger => config}/experience-cloud/sites/Logger_Test_Site.site-meta.xml (100%) rename docs/apex/{configuration => Configuration}/LoggerParameter.md (96%) create mode 100644 docs/apex/Configuration/LoggerPlugin.md rename docs/apex/{log-management => Log-Management}/LogBatchPurgeScheduler.md (100%) rename docs/apex/{log-management => Log-Management}/LogBatchPurger.md (73%) create mode 100644 docs/apex/Log-Management/LogEntryEventHandler.md rename docs/apex/{log-management => Log-Management}/LogEntryFieldSetPicklist.md (100%) rename docs/apex/{log-management => Log-Management}/LogEntryHandler.md (100%) rename docs/apex/{log-management => Log-Management}/LogEntryTagHandler.md (100%) rename docs/apex/{log-management => Log-Management}/LogHandler.md (100%) rename docs/apex/{log-management => Log-Management}/LogMassDeleteExtension.md (100%) create mode 100644 docs/apex/Log-Management/LoggerBatchableContext.md create mode 100644 docs/apex/Log-Management/LoggerEmailSender.md rename docs/apex/{configuration => Log-Management}/LoggerEmailUtils.md (100%) create mode 100644 docs/apex/Log-Management/LoggerSObjectHandler.md create mode 100644 docs/apex/Log-Management/LoggerSObjectMetadata.md rename docs/apex/{log-management => Log-Management}/LoggerSettingsController.md (97%) rename docs/apex/{log-management => Log-Management}/LoggerTagHandler.md (100%) create mode 100644 docs/apex/Log-Management/LoggerTriggerableContext.md rename docs/apex/{log-management => Log-Management}/RelatedLogEntriesController.md (100%) rename docs/apex/{logger-engine => Logger-Engine}/ComponentLogger.md (100%) rename docs/apex/{logger-engine => Logger-Engine}/FlowCollectionLogEntry.md (100%) rename docs/apex/{logger-engine => Logger-Engine}/FlowLogEntry.md (100%) rename docs/apex/{logger-engine => Logger-Engine}/FlowLogger.md (100%) rename docs/apex/{logger-engine => Logger-Engine}/FlowRecordLogEntry.md (100%) rename docs/apex/{logger-engine => Logger-Engine}/LogEntryEventBuilder.md (93%) rename docs/apex/{logger-engine => Logger-Engine}/LogMessage.md (100%) rename docs/apex/{logger-engine => Logger-Engine}/Logger.md (99%) create mode 100644 docs/apex/Logger-Engine/LoggerDataStore.md create mode 100644 docs/apex/Logger-Engine/LoggerSObjectHandler.md create mode 100644 docs/apex/Logger-Engine/LoggerTriggerableContext.md create mode 100644 docs/apex/Plugins/LogEntryArchiveBuilder.md create mode 100644 docs/apex/Plugins/LogEntryArchiveController.md create mode 100644 docs/apex/Plugins/LogEntryArchivePlugin.md create mode 100644 docs/apex/Plugins/LogRetentionRulesPlugin.md create mode 100644 docs/apex/Plugins/SlackLoggerPlugin.md create mode 100644 docs/apex/Test-Utilities/LoggerMockDataCreator.md create mode 100644 docs/apex/Test-Utilities/LoggerMockDataStore.md create mode 100644 docs/apex/Test-Utilities/LoggerTestConfigurator.md delete mode 100644 docs/apex/log-management/LogEntryEventHandler.md delete mode 100644 docs/apex/log-management/LoggerSObjectHandler.md delete mode 100644 docs/apex/log-management/LoggerSObjectMetadata.md delete mode 100644 docs/apex/plugin-framework/LoggerSObjectHandlerPlugin.md create mode 100644 nebula-logger/core/main/configuration/classes/LoggerPlugin.cls rename nebula-logger/core/main/configuration/classes/{LoggerEmailUtils.cls-meta.xml => LoggerPlugin.cls-meta.xml} (100%) create mode 100644 nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryEventHandlerEnabled.md-meta.xml create mode 100644 nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryHandlerEnabled.md-meta.xml create mode 100644 nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryTagHandlerEnabled.md-meta.xml create mode 100644 nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogHandlerEnabled.md-meta.xml create mode 100644 nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLoggerTagHandlerEnabled.md-meta.xml rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/customMetadata/LoggerSObjectHandler.LogEntryEventHandler.md-meta.xml (63%) rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/customMetadata/LoggerSObjectHandler.LogEntryHandler.md-meta.xml (63%) rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/customMetadata/LoggerSObjectHandler.LogEntryTagHandler.md-meta.xml (63%) rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/customMetadata/LoggerSObjectHandler.LogHandler.md-meta.xml (63%) rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/customMetadata/LoggerSObjectHandler.LoggerTagHandler.md-meta.xml (79%) create mode 100644 nebula-logger/core/main/configuration/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/layouts/LoggerSObjectHandler__mdt-Logger SObject Handler Layout.layout-meta.xml (77%) rename nebula-logger/core/main/{plugin-framework => configuration}/objects/LoggerPlugin__mdt/LoggerPlugin__mdt.object-meta.xml (100%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerApexClass__c.field-meta.xml rename nebula-logger/core/main/{plugin-framework/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml => configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerExecutionOrder__c.field-meta.xml} (88%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerFlowName__c.field-meta.xml rename nebula-logger/core/main/{plugin-framework => configuration}/objects/LoggerPlugin__mdt/fields/Description__c.field-meta.xml (100%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml rename nebula-logger/core/main/{plugin-framework => configuration}/objects/LoggerPlugin__mdt/fields/IsEnabled__c.field-meta.xml (100%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/Link__c.field-meta.xml create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml rename nebula-logger/core/main/{plugin-framework => configuration}/objects/LoggerPlugin__mdt/fields/PluginType__c.field-meta.xml (71%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerExecutionOrder__c.field-meta.xml create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerFlowName__c.field-meta.xml rename nebula-logger/core/main/{plugin-framework => configuration}/objects/LoggerPlugin__mdt/fields/SObjectType__c.field-meta.xml (63%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/VersionNumber__c.field-meta.xml create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/listViews/All.listView-meta.xml create mode 100644 nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/objects/LoggerSObjectHandler__mdt/LoggerSObjectHandler__mdt.object-meta.xml (73%) rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/objects/LoggerSObjectHandler__mdt/fields/IsEnabled__c.field-meta.xml (100%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/objects/LoggerSObjectHandler__mdt/fields/SObjectType__c.field-meta.xml (100%) rename nebula-logger/{managed-package/core/main/deprecated => core/main/configuration}/objects/LoggerSObjectHandler__mdt/listViews/All.listView-meta.xml (87%) create mode 100644 nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogPurgeAction__c.field-meta.xml create mode 100644 nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultPlatformEventStorageLocation__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/classes/LoggerBatchableContext.cls rename nebula-logger/core/main/log-management/classes/{LoggerSObjectHandler.cls-meta.xml => LoggerBatchableContext.cls-meta.xml} (100%) rename nebula-logger/core/main/{configuration/classes/LoggerEmailUtils.cls => log-management/classes/LoggerEmailSender.cls} (83%) rename nebula-logger/core/main/{plugin-framework/classes/LoggerSObjectHandlerPlugin.cls-meta.xml => log-management/classes/LoggerEmailSender.cls-meta.xml} (100%) delete mode 100644 nebula-logger/core/main/log-management/classes/LoggerSObjectHandler.cls rename nebula-logger/core/main/{configuration => log-management}/customPermissions/CanModifyLoggerSettings.customPermission-meta.xml (100%) rename nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/data/{getLogEntryEventSchema.json => getSchemaForName.json} (100%) delete mode 100644 nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getObjectInfo.json rename nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/{getLoggerSettingsSchema.json => getSchemaForName.json} (93%) create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/DatabaseResultCollectionSize__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBodyMasked__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBody__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestCompressed__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestEndpoint__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestMethod__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBodyMasked__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBody__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseHeaderKeys__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatusCode__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatus__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/LogEntry__c/fields/RecordCollectionSize__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/Log__c/fields/LogPurgeAction__c.field-meta.xml create mode 100644 nebula-logger/core/main/log-management/objects/Log__c/listViews/LogsToPurgeSoon.listView-meta.xml rename nebula-logger/core/main/{configuration => log-management}/permissionsets/LoggerAdmin.permissionset-meta.xml (94%) rename nebula-logger/core/main/{configuration => log-management}/permissionsets/LoggerEndUser.permissionset-meta.xml (98%) rename nebula-logger/core/main/{configuration => log-management}/permissionsets/LoggerLogCreator.permissionset-meta.xml (100%) rename nebula-logger/core/main/{configuration => log-management}/permissionsets/LoggerLogViewer.permissionset-meta.xml (94%) create mode 100644 nebula-logger/core/main/logger-engine/classes/LoggerDataStore.cls rename nebula-logger/core/{tests/common/classes/LoggerTestUtils.cls-meta.xml => main/logger-engine/classes/LoggerDataStore.cls-meta.xml} (100%) create mode 100644 nebula-logger/core/main/logger-engine/classes/LoggerSObjectHandler.cls rename nebula-logger/core/{tests/configuration/classes/LoggerEmailUtils_Tests.cls-meta.xml => main/logger-engine/classes/LoggerSObjectHandler.cls-meta.xml} (100%) create mode 100644 nebula-logger/core/main/logger-engine/classes/LoggerTriggerableContext.cls rename nebula-logger/core/{tests/plugin-framework/classes/LoggerSObjectHandlerPlugin_Tests.cls-meta.xml => main/logger-engine/classes/LoggerTriggerableContext.cls-meta.xml} (100%) create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/DatabaseResultCollectionSize__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBodyMasked__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBody__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestCompressed__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestEndpoint__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestMethod__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBodyMasked__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBody__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseHeaderKeys__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatusCode__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatus__c.field-meta.xml create mode 100644 nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/RecordCollectionSize__c.field-meta.xml delete mode 100644 nebula-logger/core/main/plugin-framework/classes/LoggerSObjectHandlerPlugin.cls delete mode 100644 nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml delete mode 100644 nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml create mode 100644 nebula-logger/core/tests/LoggerCore.testSuite-meta.xml delete mode 100644 nebula-logger/core/tests/common/classes/LoggerTestUtils.cls delete mode 100644 nebula-logger/core/tests/configuration/classes/LoggerEmailUtils_Tests.cls create mode 100644 nebula-logger/core/tests/configuration/classes/LoggerPlugin_Tests.cls rename nebula-logger/{extra-tests/tests/LogHandler_Tests_Flow.cls-meta.xml => core/tests/configuration/classes/LoggerPlugin_Tests.cls-meta.xml} (100%) create mode 100644 nebula-logger/core/tests/configuration/utilities/LoggerMockDataCreator.cls rename nebula-logger/{plugins/Slack/plugin/slack/classes/SlackLoggerPlugin.cls-meta.xml => core/tests/configuration/utilities/LoggerMockDataCreator.cls-meta.xml} (100%) create mode 100644 nebula-logger/core/tests/log-management/classes/LoggerBatchableContext_Tests.cls rename nebula-logger/{plugins/Slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls-meta.xml => core/tests/log-management/classes/LoggerBatchableContext_Tests.cls-meta.xml} (100%) create mode 100644 nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls create mode 100644 nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls-meta.xml create mode 100644 nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls create mode 100644 nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls-meta.xml create mode 100644 nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls create mode 100644 nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls-meta.xml create mode 100644 nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls create mode 100644 nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls-meta.xml create mode 100644 nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls create mode 100644 nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls-meta.xml create mode 100644 nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls create mode 100644 nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls-meta.xml delete mode 100644 nebula-logger/core/tests/plugin-framework/classes/LoggerSObjectHandlerPlugin_Tests.cls delete mode 100644 nebula-logger/core/tests/plugin-framework/testSuites/LoggerPluginFramework.testSuite-meta.xml delete mode 100644 nebula-logger/extra-tests/flows/LogHandler_Tests_Flow.flow-meta.xml create mode 100644 nebula-logger/extra-tests/flows/MockLogBatchPurgerPlugin.flow-meta.xml create mode 100644 nebula-logger/extra-tests/flows/MockLoggerSObjectHandlerPlugin.flow-meta.xml create mode 100644 nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls create mode 100644 nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls-meta.xml create mode 100644 nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls create mode 100644 nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls-meta.xml delete mode 100644 nebula-logger/extra-tests/tests/LogHandler_Tests_Flow.cls create mode 100644 nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls create mode 100644 nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls-meta.xml rename nebula-logger/{ => managed-package}/core/main/configuration/objects/LoggerSettings__c/fields/IsPlatformEventStorageEnabled__c.field-meta.xml (81%) create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls-meta.xml create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls-meta.xml create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls-meta.xml create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls-meta.xml create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls create mode 100644 nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls-meta.xml create mode 100644 nebula-logger/plugins/.images/btn-install-unlocked-package-plugin-sandbox.png rename nebula-logger/plugins/{Logger-Admin-Dashboard/images => .images}/btn-install-unlocked-package-plugin.png (100%) delete mode 100644 nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls delete mode 100644 nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/.images/log-entry-archives-tab.png create mode 100644 nebula-logger/plugins/big-object-archiving/README.md create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomLogPurgeActionArchive.md-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectImmediate.md-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectQueueable.md-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomStorageLocationBigObject.md-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerPlugin.LogEntryArchive.md-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/data/LoggerSObjectMetadata.getSchemaForName.json create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/logEntryArchives.test.js create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.html create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/LogEntryArchive__b.object-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseNumber__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseVersion__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiVersion__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchiveRetentionDate__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedById__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedByUsername__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedDate__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedById__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedByUsername__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedDate__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Comments__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ComponentType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionSize__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultJson__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EpochTimestamp__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EventUuid__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionMessage__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionStackTrace__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBodyMasked__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBody__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestCompressed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestEndpoint__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestMethod__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBodyMasked__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBody__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseHeaderKeys__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatusCode__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatus__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsClosed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsResolved__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Issue__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueryMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesMax__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesUsed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Locale__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogEntryName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogPurgeAction__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogRetentionDate__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedById__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedByUsername__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggerVersionNumber__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevelOrdinal__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevel__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginApplication__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginBrowser__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginDomain__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginHistoryId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginPlatform__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogoutUrl__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageMasked__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageTruncated__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Message__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLoginUrl__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLogoutUrl__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkSelfRegistrationUrl__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkUrlPathPrefix__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationDomainUrl__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationEnvironmentType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationNamespacePrefix__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginLocation__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ParentLogTransactionId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Priority__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionSize__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJsonMasked__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJson__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectClassification__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectTypeNamespace__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Scenario__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionSecurityLevel__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SourceIp__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/StackTrace__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Status__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SystemMode__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Tags__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ThemeDisplayed__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimestampString__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Timestamp__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionEntryNumber__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerIsExecuting__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerOperationType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerSObjectType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseDefinitionKey__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevelOrdinal__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevel__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleId__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleName__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserType__c.field-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/indexes/LogEntryArchiveIndex.indexe-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/permissionsets/LoggerLogEntryArchiveAdmin.permissionset-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/tabs/LogEntryArchives.tab-meta.xml create mode 100644 nebula-logger/plugins/big-object-archiving/plugin/testSuites/LoggerLogEntryArchivePlugin.testSuite-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/.images/example-log-retention-rule-with-conditions.png create mode 100644 nebula-logger/plugins/log-retention-rules/.images/example-log-retention-rules-list-view.png create mode 100644 nebula-logger/plugins/log-retention-rules/README.md create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Error_Logs.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Scenarios_With_Errors.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_2_Additional_Error_Logs.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Error_Condition.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_A_Condition.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_B_Condition.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_condition.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_2_Error_Condition.md-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LoggerPlugin.LogRetentionRules.md-meta.xml rename nebula-logger/{core/main/plugin-framework/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml => plugins/log-retention-rules/plugin/layouts/LogRetentionRuleCondition__mdt-Log Retention Rule Condition Layout.layout-meta.xml} (82%) create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/layouts/LogRetentionRule__mdt-Log Retention Rule Layout.layout-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/LogRetentionRuleCondition__mdt.object-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/FieldPath__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/LogRetentionRule__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Operator__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/SortOrder__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/ValueType__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Value__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/listViews/All.listView-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/LogRetentionRule__mdt.object-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ConditionLogicType__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/CustomConditionLogic__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ExecutionOrder__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/IsEnabled__c.field-meta.xml create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/NumberOfDaysToRetainLogs__c.field-meta.xml rename nebula-logger/{core/main/plugin-framework/objects/LoggerPlugin__mdt => plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt}/listViews/All.listView-meta.xml (69%) create mode 100644 nebula-logger/plugins/log-retention-rules/plugin/testSuites/LoggerLogRetentionRulesPlugin.testSuite-meta.xml rename nebula-logger/plugins/{Slack/images => logger-admin-dashboard/.images}/btn-install-unlocked-package-plugin.png (100%) rename nebula-logger/plugins/{Logger-Admin-Dashboard/images => logger-admin-dashboard/.images}/logger-admin-dashboard.png (100%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/README.md (64%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/plugin/dashboards/LogDashboards.dashboardFolder-meta.xml (100%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/plugin/dashboards/LogDashboards/LoggerAdmin.dashboard-meta.xml (99%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/plugin/reports/LogReports/LogEntryDailyRetentionSummary.report-meta.xml (100%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/plugin/reports/LogReports/LogEntryDailySummary.report-meta.xml (100%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/plugin/reports/LogReports/LogEntryOriginSummary.report-meta.xml (100%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/plugin/reports/LogReports/LogEntryScenarioSummary.report-meta.xml (100%) rename nebula-logger/plugins/{Logger-Admin-Dashboard => logger-admin-dashboard}/plugin/reports/LogReports/LogEntrySummary.report-meta.xml (100%) rename nebula-logger/plugins/{Slack/images => slack/.images}/slack-plugin-notification.png (100%) rename nebula-logger/plugins/{Slack/images => slack/.images}/slack-plugin-parameters.png (100%) rename nebula-logger/plugins/{Slack => slack}/README.md (79%) rename nebula-logger/plugins/{Slack => slack}/plugin/core/main/README.md (100%) rename nebula-logger/plugins/{Slack => slack}/plugin/core/main/log-management/objects/Log__c/fields/SendSlackNotification__c.field-meta.xml (100%) rename nebula-logger/plugins/{Slack => slack}/plugin/core/main/log-management/objects/Log__c/fields/SlackNotificationDate__c.field-meta.xml (100%) rename nebula-logger/plugins/{Slack => slack}/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsSentToSlack.listView-meta.xml (100%) rename nebula-logger/plugins/{Slack => slack}/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsToBeSentToSlack.listView-meta.xml (100%) rename nebula-logger/plugins/{Slack => slack}/plugin/slack/classes/SlackLoggerPlugin.cls (77%) create mode 100644 nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin.cls-meta.xml create mode 100644 nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls create mode 100644 nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls-meta.xml rename nebula-logger/plugins/{Slack => slack}/plugin/slack/customMetadata/LoggerParameter.SlackEndpoint.md-meta.xml (100%) rename nebula-logger/plugins/{Slack => slack}/plugin/slack/customMetadata/LoggerParameter.SlackNotificationLoggingLevel.md-meta.xml (100%) create mode 100644 nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml rename nebula-logger/plugins/{Slack => slack}/plugin/slack/permissionsets/LoggerSlackPluginAdmin.permissionset-meta.xml (100%) rename nebula-logger/plugins/{Slack => slack}/plugin/slack/remoteSiteSettings/Slack.remoteSite-meta.xml (100%) create mode 100644 nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls create mode 100644 nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls-meta.xml create mode 100644 scripts/data/create-sample-log-entry-archives.apex create mode 100644 scripts/data/query-log-entry-archive-big-object.apex diff --git a/.forceignore b/.forceignore index 637ef25a3..d12b69942 100644 --- a/.forceignore +++ b/.forceignore @@ -4,7 +4,7 @@ nebula-logger/**/main/default/ nebula-logger/managed-package/sfdx-project.json -nebula-logger/managed-package/**/testSuites/** +nebula-logger/managed-package/**/*.testSuite-meta.xml # Directory/package-specific README files **/README.md diff --git a/.github/codecov.yml b/.github/codecov.yml index 0d7eb5e21..a8d661d84 100644 --- a/.github/codecov.yml +++ b/.github/codecov.yml @@ -7,6 +7,6 @@ coverage: if_ci_failed: success patch: off ignore: - - 'nebula-logger-recipes/**/*' + - 'nebula-logger/recipes/**/*' comment: behavior: new diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 2f9a39e82..d0e8d514f 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -160,8 +160,8 @@ jobs: - name: 'Create Base Scratch Org' run: npx sfdx force:org:create --durationdays 1 --definitionfile ./config/scratch-orgs/base-scratch-def.json --wait 20 --setdefaultusername --json - - name: 'Push Source to Scratch Org' - run: npm run source:push + - name: 'Deploy Source to Scratch Org' + run: npm run source:deploy - name: 'Assign Logger Admin Permission Set' run: npm run permset:assign:admin @@ -213,8 +213,8 @@ jobs: - name: 'Deploy Test Experience Site Metadata' run: npm run experience:deploy - - name: 'Push Source to Scratch Org' - run: npm run source:push + - name: 'Deploy Source to Scratch Org' + run: npm run source:deploy - name: 'Assign Logger Admin Permission Set' run: npm run permset:assign:admin diff --git a/README.md b/README.md index e0ac9bb3b..8fa307f18 100644 --- a/README.md +++ b/README.md @@ -5,10 +5,10 @@ Designed for Salesforce admins, developers & architects. A robust logger for Apex, Lightning Components, Flow, Process Builder & Integrations. -## Unlocked Package - v4.7.0 +## Unlocked Package - v4.7.1 -[![Install Unlocked Package in a Sandbox](./images/btn-install-unlocked-package-sandbox.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015lXSQAY) -[![Install Unlocked Package in Production](./images/btn-install-unlocked-package-production.png)](https://login.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015lXSQAY) +[![Install Unlocked Package in a Sandbox](./images/btn-install-unlocked-package-sandbox.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015lgBQAQ) +[![Install Unlocked Package in Production](./images/btn-install-unlocked-package-production.png)](https://login.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015lgBQAQ) [![View Documentation](./images/btn-view-documentation.png)](https://jongpie.github.io/NebulaLogger/) ## Managed Package - v4.7.0 @@ -702,9 +702,9 @@ The optional [Slack plugin](./nebula-logger-plugins/Slack/) leverages the Nebula - Extend permission sets to include field-level security for your custom fields - Leverage the new `LoggerParameter__mdt` CMDT object to store configuration for your plugin -Check out the [Slack plugin](./nebula-logger-plugins/Slack/) for more details on how to install & customize the plugin +Check out the [Slack plugin](./nebula-logger-plugins/slack/) for more details on how to install & customize the plugin -![Slack plugin: notification](./nebula-logger-plugins/Slack/images/slack-plugin-notification.png) +![Slack plugin: notification](./nebula-logger/plugins/slack/.images/slack-plugin-notification.png) --- diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1.site-meta.xml b/config/experience-cloud/experiences/Logger_Test_Site1.site-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1.site-meta.xml rename to config/experience-cloud/experiences/Logger_Test_Site1.site-meta.xml diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/brandingSets/buildYourOwnLWR.json b/config/experience-cloud/experiences/Logger_Test_Site1/brandingSets/buildYourOwnLWR.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/brandingSets/buildYourOwnLWR.json rename to config/experience-cloud/experiences/Logger_Test_Site1/brandingSets/buildYourOwnLWR.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/config/languages.json b/config/experience-cloud/experiences/Logger_Test_Site1/config/languages.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/config/languages.json rename to config/experience-cloud/experiences/Logger_Test_Site1/config/languages.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/config/loggerTestSite.json b/config/experience-cloud/experiences/Logger_Test_Site1/config/loggerTestSite.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/config/loggerTestSite.json rename to config/experience-cloud/experiences/Logger_Test_Site1/config/loggerTestSite.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/config/mainAppPage.json b/config/experience-cloud/experiences/Logger_Test_Site1/config/mainAppPage.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/config/mainAppPage.json rename to config/experience-cloud/experiences/Logger_Test_Site1/config/mainAppPage.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/checkPassword.json b/config/experience-cloud/experiences/Logger_Test_Site1/routes/checkPassword.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/checkPassword.json rename to config/experience-cloud/experiences/Logger_Test_Site1/routes/checkPassword.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/error.json b/config/experience-cloud/experiences/Logger_Test_Site1/routes/error.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/error.json rename to config/experience-cloud/experiences/Logger_Test_Site1/routes/error.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/forgotPassword.json b/config/experience-cloud/experiences/Logger_Test_Site1/routes/forgotPassword.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/forgotPassword.json rename to config/experience-cloud/experiences/Logger_Test_Site1/routes/forgotPassword.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/home.json b/config/experience-cloud/experiences/Logger_Test_Site1/routes/home.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/home.json rename to config/experience-cloud/experiences/Logger_Test_Site1/routes/home.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/login.json b/config/experience-cloud/experiences/Logger_Test_Site1/routes/login.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/login.json rename to config/experience-cloud/experiences/Logger_Test_Site1/routes/login.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/register.json b/config/experience-cloud/experiences/Logger_Test_Site1/routes/register.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/register.json rename to config/experience-cloud/experiences/Logger_Test_Site1/routes/register.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/serviceNotAvailable.json b/config/experience-cloud/experiences/Logger_Test_Site1/routes/serviceNotAvailable.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/routes/serviceNotAvailable.json rename to config/experience-cloud/experiences/Logger_Test_Site1/routes/serviceNotAvailable.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/themes/buildYourOwnLWR.json b/config/experience-cloud/experiences/Logger_Test_Site1/themes/buildYourOwnLWR.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/themes/buildYourOwnLWR.json rename to config/experience-cloud/experiences/Logger_Test_Site1/themes/buildYourOwnLWR.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/checkPassword.json b/config/experience-cloud/experiences/Logger_Test_Site1/views/checkPassword.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/checkPassword.json rename to config/experience-cloud/experiences/Logger_Test_Site1/views/checkPassword.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/error.json b/config/experience-cloud/experiences/Logger_Test_Site1/views/error.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/error.json rename to config/experience-cloud/experiences/Logger_Test_Site1/views/error.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/forgotPassword.json b/config/experience-cloud/experiences/Logger_Test_Site1/views/forgotPassword.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/forgotPassword.json rename to config/experience-cloud/experiences/Logger_Test_Site1/views/forgotPassword.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/home.json b/config/experience-cloud/experiences/Logger_Test_Site1/views/home.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/home.json rename to config/experience-cloud/experiences/Logger_Test_Site1/views/home.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/login.json b/config/experience-cloud/experiences/Logger_Test_Site1/views/login.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/login.json rename to config/experience-cloud/experiences/Logger_Test_Site1/views/login.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/register.json b/config/experience-cloud/experiences/Logger_Test_Site1/views/register.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/register.json rename to config/experience-cloud/experiences/Logger_Test_Site1/views/register.json diff --git a/nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/serviceNotAvailable.json b/config/experience-cloud/experiences/Logger_Test_Site1/views/serviceNotAvailable.json similarity index 100% rename from nebula-logger/experience-cloud/experiences/Logger_Test_Site1/views/serviceNotAvailable.json rename to config/experience-cloud/experiences/Logger_Test_Site1/views/serviceNotAvailable.json diff --git a/nebula-logger/experience-cloud/networks/Logger Test Site.network-meta.xml b/config/experience-cloud/networks/Logger Test Site.network-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/networks/Logger Test Site.network-meta.xml rename to config/experience-cloud/networks/Logger Test Site.network-meta.xml diff --git a/nebula-logger/experience-cloud/pages/BandwidthExceeded.page b/config/experience-cloud/pages/BandwidthExceeded.page similarity index 100% rename from nebula-logger/experience-cloud/pages/BandwidthExceeded.page rename to config/experience-cloud/pages/BandwidthExceeded.page diff --git a/nebula-logger/experience-cloud/pages/BandwidthExceeded.page-meta.xml b/config/experience-cloud/pages/BandwidthExceeded.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/BandwidthExceeded.page-meta.xml rename to config/experience-cloud/pages/BandwidthExceeded.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/ChangePassword.page b/config/experience-cloud/pages/ChangePassword.page similarity index 100% rename from nebula-logger/experience-cloud/pages/ChangePassword.page rename to config/experience-cloud/pages/ChangePassword.page diff --git a/nebula-logger/experience-cloud/pages/ChangePassword.page-meta.xml b/config/experience-cloud/pages/ChangePassword.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/ChangePassword.page-meta.xml rename to config/experience-cloud/pages/ChangePassword.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/CommunitiesLanding.page b/config/experience-cloud/pages/CommunitiesLanding.page similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesLanding.page rename to config/experience-cloud/pages/CommunitiesLanding.page diff --git a/nebula-logger/experience-cloud/pages/CommunitiesLanding.page-meta.xml b/config/experience-cloud/pages/CommunitiesLanding.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesLanding.page-meta.xml rename to config/experience-cloud/pages/CommunitiesLanding.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/CommunitiesLogin.page b/config/experience-cloud/pages/CommunitiesLogin.page similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesLogin.page rename to config/experience-cloud/pages/CommunitiesLogin.page diff --git a/nebula-logger/experience-cloud/pages/CommunitiesLogin.page-meta.xml b/config/experience-cloud/pages/CommunitiesLogin.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesLogin.page-meta.xml rename to config/experience-cloud/pages/CommunitiesLogin.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/CommunitiesSelfReg.page b/config/experience-cloud/pages/CommunitiesSelfReg.page similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesSelfReg.page rename to config/experience-cloud/pages/CommunitiesSelfReg.page diff --git a/nebula-logger/experience-cloud/pages/CommunitiesSelfReg.page-meta.xml b/config/experience-cloud/pages/CommunitiesSelfReg.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesSelfReg.page-meta.xml rename to config/experience-cloud/pages/CommunitiesSelfReg.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/CommunitiesSelfRegConfirm.page b/config/experience-cloud/pages/CommunitiesSelfRegConfirm.page similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesSelfRegConfirm.page rename to config/experience-cloud/pages/CommunitiesSelfRegConfirm.page diff --git a/nebula-logger/experience-cloud/pages/CommunitiesSelfRegConfirm.page-meta.xml b/config/experience-cloud/pages/CommunitiesSelfRegConfirm.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/CommunitiesSelfRegConfirm.page-meta.xml rename to config/experience-cloud/pages/CommunitiesSelfRegConfirm.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/Exception.page b/config/experience-cloud/pages/Exception.page similarity index 100% rename from nebula-logger/experience-cloud/pages/Exception.page rename to config/experience-cloud/pages/Exception.page diff --git a/nebula-logger/experience-cloud/pages/Exception.page-meta.xml b/config/experience-cloud/pages/Exception.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/Exception.page-meta.xml rename to config/experience-cloud/pages/Exception.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/FileNotFound.page b/config/experience-cloud/pages/FileNotFound.page similarity index 100% rename from nebula-logger/experience-cloud/pages/FileNotFound.page rename to config/experience-cloud/pages/FileNotFound.page diff --git a/nebula-logger/experience-cloud/pages/FileNotFound.page-meta.xml b/config/experience-cloud/pages/FileNotFound.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/FileNotFound.page-meta.xml rename to config/experience-cloud/pages/FileNotFound.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/ForgotPasswordConfirm.page b/config/experience-cloud/pages/ForgotPasswordConfirm.page similarity index 100% rename from nebula-logger/experience-cloud/pages/ForgotPasswordConfirm.page rename to config/experience-cloud/pages/ForgotPasswordConfirm.page diff --git a/nebula-logger/experience-cloud/pages/ForgotPasswordConfirm.page-meta.xml b/config/experience-cloud/pages/ForgotPasswordConfirm.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/ForgotPasswordConfirm.page-meta.xml rename to config/experience-cloud/pages/ForgotPasswordConfirm.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/InMaintenance.page b/config/experience-cloud/pages/InMaintenance.page similarity index 100% rename from nebula-logger/experience-cloud/pages/InMaintenance.page rename to config/experience-cloud/pages/InMaintenance.page diff --git a/nebula-logger/experience-cloud/pages/InMaintenance.page-meta.xml b/config/experience-cloud/pages/InMaintenance.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/InMaintenance.page-meta.xml rename to config/experience-cloud/pages/InMaintenance.page-meta.xml diff --git a/nebula-logger/experience-cloud/pages/SiteRegisterConfirm.page b/config/experience-cloud/pages/SiteRegisterConfirm.page similarity index 100% rename from nebula-logger/experience-cloud/pages/SiteRegisterConfirm.page rename to config/experience-cloud/pages/SiteRegisterConfirm.page diff --git a/nebula-logger/experience-cloud/pages/SiteRegisterConfirm.page-meta.xml b/config/experience-cloud/pages/SiteRegisterConfirm.page-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/pages/SiteRegisterConfirm.page-meta.xml rename to config/experience-cloud/pages/SiteRegisterConfirm.page-meta.xml diff --git a/nebula-logger/experience-cloud/profiles/Logger Test Site Profile.profile-meta.xml b/config/experience-cloud/profiles/Logger Test Site Profile.profile-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/profiles/Logger Test Site Profile.profile-meta.xml rename to config/experience-cloud/profiles/Logger Test Site Profile.profile-meta.xml diff --git a/nebula-logger/experience-cloud/sites/Logger_Test_Site.site-meta.xml b/config/experience-cloud/sites/Logger_Test_Site.site-meta.xml similarity index 100% rename from nebula-logger/experience-cloud/sites/Logger_Test_Site.site-meta.xml rename to config/experience-cloud/sites/Logger_Test_Site.site-meta.xml diff --git a/config/scratch-orgs/base-scratch-def.json b/config/scratch-orgs/base-scratch-def.json index ce7b50fd4..f565da5ed 100644 --- a/config/scratch-orgs/base-scratch-def.json +++ b/config/scratch-orgs/base-scratch-def.json @@ -12,6 +12,9 @@ "communitiesSettings": { "enableNetworksEnabled": false }, + "pathAssistantSettings": { + "pathAssistantEnabled": true + }, "userManagementSettings": { "enableEnhancedPermsetMgmt": true, "enableEnhancedProfileMgmt": true, diff --git a/config/scratch-orgs/experience-cloud-scratch-def.json b/config/scratch-orgs/experience-cloud-scratch-def.json index 86f826db1..e68b9efaf 100644 --- a/config/scratch-orgs/experience-cloud-scratch-def.json +++ b/config/scratch-orgs/experience-cloud-scratch-def.json @@ -15,6 +15,9 @@ "experienceBundleSettings": { "enableExperienceBundleMetadata": true }, + "pathAssistantSettings": { + "pathAssistantEnabled": true + }, "userManagementSettings": { "enableEnhancedPermsetMgmt": true, "enableEnhancedProfileMgmt": true, diff --git a/docs/apex/configuration/LoggerParameter.md b/docs/apex/Configuration/LoggerParameter.md similarity index 96% rename from docs/apex/configuration/LoggerParameter.md rename to docs/apex/Configuration/LoggerParameter.md index 7cd4c46aa..f758d5de1 100644 --- a/docs/apex/configuration/LoggerParameter.md +++ b/docs/apex/Configuration/LoggerParameter.md @@ -458,4 +458,24 @@ List<String> A list that's retrieved using the parameter if it's not null, otherwise the default list is returned. +#### `matchOnPrefix(String developerNamePrefix)` → `List` + +matchOnPrefix description + +##### Parameters + +| Param | Description | +| --------------------- | ------------------------------------------------------------------------------------------- | +| `developerNamePrefix` | A prefix that has been used in the `DeveloperName` for multiple `LoggerParameter_t` records | + +##### Return + +**Type** + +List<LoggerParameter_t> + +**Description** + +The list of matching `LoggerParameter_t` records + --- diff --git a/docs/apex/Configuration/LoggerPlugin.md b/docs/apex/Configuration/LoggerPlugin.md new file mode 100644 index 000000000..7473b9b6e --- /dev/null +++ b/docs/apex/Configuration/LoggerPlugin.md @@ -0,0 +1,116 @@ +--- +layout: default +--- + +## LoggerPlugin class + +The core of the plugin framework, used to create custom Apex & Flow plugins for `LoggerSObjectHandler` and `LogBatchPurger` based on configurations stored in the custom metadata type `LoggerPlugin_t` + +--- + +### Properties + +#### `pluginConfiguration` → `LoggerPlugin_t` + +--- + +### Methods + +#### `PluginConfigurationSorter(LoggerPlugin_t pluginConfiguration)` → `public` + +#### `compareTo(Object compareTo)` → `Integer` + +#### `getFilteredPluginConfigurations(List populatedFilterFields, Schema.SObjectField sortByField)` → `List` + +Filters the configured `LoggerPlugin_t` records based on a list of `SObjectField` - only records that have a value for 1 or more of the specified `populatedFilterFields` will be returned, sorted by the specified `SObjectField` parameter `sortByField` + +##### Parameters + +| Param | Description | +| ----------------------- | ----------------------------------------------------------------------------------------------------------------------------- | +| `populatedFilterFields` | The list of `SObjectField` to check on each `LoggerPlugin_t` record - filtering logic checks for a non-null value | +| `sortByField` | The `SObjectField` to use to sort the list of matches. The method also uses `DeveloperName` as a secondary field for sorting. | + +##### Return + +**Type** + +List<LoggerPlugin_t> + +**Description** + +The list of matching `LoggerPlugin_t` records + +#### `newBatchableInstance(String apexClassTypeName)` → `Batchable` + +Creates an instance of the class `LoggerPlugin.Batchable` based on the provided `LoggerPlugin_t` configuration + +##### Parameters + +| Param | Description | +| ------------------- | ------------------------------------------------------------------- | +| `apexClassTypeName` | The name of the Apex class that implements `LoggerPlugin.Batchable` | + +##### Return + +**Type** + +Batchable + +**Description** + +The dynamically created instance of `LoggerPlugin.Batchable`, + +#### `newTriggerableInstance(String apexClassTypeName)` → `Triggerable` + +Creates an instance of the class `LoggerPlugin.Triggerable` based on the provided `LoggerPlugin_t` configuration + +##### Parameters + +| Param | Description | +| ------------------- | --------------------------------------------------------------------- | +| `apexClassTypeName` | The name of the Apex class that implements `LoggerPlugin.Triggerable` | + +##### Return + +**Type** + +Triggerable + +**Description** + +The dynamically created instance of `LoggerPlugin.Triggerable`, + +#### `sortBy(Schema.SObjectField field)` → `PluginConfigurationSorter` + +--- + +### Inner Classes + +#### LoggerPlugin.Batchable interface + +Interface used to create plugins that can be used within Logger's batch job `LogBatchPurger` + +--- + +##### Methods + +###### `execute(LoggerPlugin_t configuration, LoggerBatchableContext input, List scopeRecords)` → `void` + +###### `finish(LoggerPlugin_t configuration, LoggerBatchableContext input)` → `void` + +###### `start(LoggerPlugin_t configuration, LoggerBatchableContext input)` → `void` + +--- + +#### LoggerPlugin.Triggerable interface + +Interface used to create plugins that can be used within Logger's trigger handler framework `LoggerSObjectHandler` + +--- + +##### Methods + +###### `execute(LoggerPlugin_t configuration, LoggerTriggerableContext input)` → `void` + +--- diff --git a/docs/apex/log-management/LogBatchPurgeScheduler.md b/docs/apex/Log-Management/LogBatchPurgeScheduler.md similarity index 100% rename from docs/apex/log-management/LogBatchPurgeScheduler.md rename to docs/apex/Log-Management/LogBatchPurgeScheduler.md diff --git a/docs/apex/log-management/LogBatchPurger.md b/docs/apex/Log-Management/LogBatchPurger.md similarity index 73% rename from docs/apex/log-management/LogBatchPurger.md rename to docs/apex/Log-Management/LogBatchPurger.md index b735cfd35..6b4109924 100644 --- a/docs/apex/log-management/LogBatchPurger.md +++ b/docs/apex/Log-Management/LogBatchPurger.md @@ -31,7 +31,7 @@ Required by the Database.Batchable interface, this method executes the logic for | `batchableContext` | - The context of the current batch job. | | `logRecords` | - The log records to purge. | -#### `execute(Database.BatchableContext batchableContext, List loggerRecords)` → `void` +#### `execute(Database.BatchableContext batchableContext, List scopeRecords)` → `void` Required by the Database.Batchable interface, this method executes the logic for purging log records. @@ -40,7 +40,7 @@ Required by the Database.Batchable interface, this method executes the logic for | Param | Description | | ------------------ | --------------------------------------- | | `batchableContext` | - The context of the current batch job. | -| `loggerRecords` | - The log records to purge. | +| `scopeRecords` | - The log records to purge. | #### `finish(Database.BatchableContext batchableContext)` → `void` @@ -52,6 +52,26 @@ Required by the Database.Batchable interface. This method runs after all batch j | ------------------ | ------------------------------- | | `batchableContext` | - The context of the batch jobs | +#### `setChainedBatchSize(Integer chainedBatchSize)` → `LogBatchPurger` + +The `LogBatchPurger` job is designed to run several instances - typically, it runs on `LogEntryTag__c`, then `LogEntry__c`, and finally `Log__c`. This method provides a way to control the batch size used for the chained instances of `LogBachPurger` + +##### Parameters + +| Param | Description | +| ------------------ | ------------------------------------------------------------------------------ | +| `chainedBatchSize` | The batch size to use for any subsequent chained instances of `LogBatchPurger` | + +##### Return + +**Type** + +LogBatchPurger + +**Description** + +The same instance of `LogBatchPurger`, useful for chaining methods + #### `start(Database.BatchableContext batchableContext)` → `Database.QueryLocator` Required by the Database.Batchable interface. Collects the records / objects passed in to the batch instance and returns a Databae.QueryLocator reference representing the current iteration. diff --git a/docs/apex/Log-Management/LogEntryEventHandler.md b/docs/apex/Log-Management/LogEntryEventHandler.md new file mode 100644 index 000000000..15e6b8279 --- /dev/null +++ b/docs/apex/Log-Management/LogEntryEventHandler.md @@ -0,0 +1,45 @@ +--- +layout: default +--- + +## LogEntryEventHandler class + +Processes `LogEntryEvent__e` platform events and normalizes the data into `Log__c` and `LogEntry__c` records + +--- + +### Constructors + +#### `LogEntryEventHandler()` + +## Default constructor, used by the trigger `LogEntryEvent.trigger` + +### Properties + +#### `releaseNumber` → `String` + +String containing the release number. + +#### `releaseVersion` → `String` + +String containing the release version. + +--- + +### Methods + +#### `getSObjectType()` → `SObjectType` + +Returns SObject Type that the handler is responsible for processing + +##### Return + +**Type** + +SObjectType + +**Description** + +The instance of `SObjectType` + +--- diff --git a/docs/apex/log-management/LogEntryFieldSetPicklist.md b/docs/apex/Log-Management/LogEntryFieldSetPicklist.md similarity index 100% rename from docs/apex/log-management/LogEntryFieldSetPicklist.md rename to docs/apex/Log-Management/LogEntryFieldSetPicklist.md diff --git a/docs/apex/log-management/LogEntryHandler.md b/docs/apex/Log-Management/LogEntryHandler.md similarity index 100% rename from docs/apex/log-management/LogEntryHandler.md rename to docs/apex/Log-Management/LogEntryHandler.md diff --git a/docs/apex/log-management/LogEntryTagHandler.md b/docs/apex/Log-Management/LogEntryTagHandler.md similarity index 100% rename from docs/apex/log-management/LogEntryTagHandler.md rename to docs/apex/Log-Management/LogEntryTagHandler.md diff --git a/docs/apex/log-management/LogHandler.md b/docs/apex/Log-Management/LogHandler.md similarity index 100% rename from docs/apex/log-management/LogHandler.md rename to docs/apex/Log-Management/LogHandler.md diff --git a/docs/apex/log-management/LogMassDeleteExtension.md b/docs/apex/Log-Management/LogMassDeleteExtension.md similarity index 100% rename from docs/apex/log-management/LogMassDeleteExtension.md rename to docs/apex/Log-Management/LogMassDeleteExtension.md diff --git a/docs/apex/Log-Management/LoggerBatchableContext.md b/docs/apex/Log-Management/LoggerBatchableContext.md new file mode 100644 index 000000000..9a2b9661d --- /dev/null +++ b/docs/apex/Log-Management/LoggerBatchableContext.md @@ -0,0 +1,40 @@ +--- +layout: default +--- + +## LoggerBatchableContext class + +Class used by the logging system for batch contextual details + +### Related + +[LogBatchPurger](LogBatchPurger) + +[LoggerPlugin](LoggerPlugin) + +--- + +### Constructors + +#### `LoggerBatchableContext(Database.BatchableContext batchableContext, Schema.SObjectType sobjectType)` + +Constructor used to set the 2 properties + +##### Parameters + +| Param | Description | +| ------------------ | ------------------------------------------------------------------------------------------------- | +| `batchableContext` | The instance of `Database.BatchableContextbatchableContext`, provided by the platform at run-time | +| `sobjectType` | The `SObjectType` that will be queried & purged | + +--- + +### Properties + +#### `batchableContext` → `Database.BatchableContext` + +#### `sobjectType` → `Schema.SObjectType` + +#### `sobjectTypeName` → `String` + +--- diff --git a/docs/apex/Log-Management/LoggerEmailSender.md b/docs/apex/Log-Management/LoggerEmailSender.md new file mode 100644 index 000000000..4a37e8515 --- /dev/null +++ b/docs/apex/Log-Management/LoggerEmailSender.md @@ -0,0 +1,35 @@ +--- +layout: default +--- + +## LoggerEmailSender class + +Builds and sends email notifications when internal exceptions occur within the logging system + +--- + +### Methods + +#### `sendErrorEmail(Schema.SObjectType sobjectType, List saveResults)` → `void` + +Sends an error email notification to the org's list of Apex Exception Email recipients, configured under Setup --> Email --> Apex Exception Email + +##### Parameters + +| Param | Description | +| ------------- | -------------------------------------------------------------- | +| `sobjectType` | The SObjectType of records being saved. | +| `saveResults` | The list of Database.SaveResult instances to use in the email. | + +#### `sendErrorEmail(Schema.SObjectType sobjectType, List upsertResults)` → `void` + +Sends an error email notification to the org's list of Apex Exception Email recipients, configured under Setup --> Email --> Apex Exception Email + +##### Parameters + +| Param | Description | +| --------------- | ---------------------------------------------------------------- | +| `sobjectType` | The SObjectType of records being saved. | +| `upsertResults` | The list of Database.UpsertResult instances to use in the email. | + +--- diff --git a/docs/apex/configuration/LoggerEmailUtils.md b/docs/apex/Log-Management/LoggerEmailUtils.md similarity index 100% rename from docs/apex/configuration/LoggerEmailUtils.md rename to docs/apex/Log-Management/LoggerEmailUtils.md diff --git a/docs/apex/Log-Management/LoggerSObjectHandler.md b/docs/apex/Log-Management/LoggerSObjectHandler.md new file mode 100644 index 000000000..e0cce2f10 --- /dev/null +++ b/docs/apex/Log-Management/LoggerSObjectHandler.md @@ -0,0 +1,51 @@ +--- +layout: default +--- + +## LoggerSObjectHandler class + +Abstract class used by trigger handlers for shared logic + +--- + +### Constructors + +#### `LoggerSObjectHandler()` + +## Default constructor + +### Methods + +#### `execute()` → `void` + +Runs the handler class's logic, as well as any configured plugins + +#### `getHandlerControlParameterName()` → `String` + +Returns the string value of the `LoggerParameter_t` record that controls if the handler is enabled. The `LoggerSObjectHandler` class uses this method to retrieve the corresponding `LoggerParameter_t` to determine if the class should execute. + +##### Return + +**Type** + +String + +**Description** + +The `DeveloperName` value of the `LoggerParameter_t` that controls if the handler is enabled + +#### `getSObjectType()` → `Schema.SObjectType` + +Returns the SObject Type that the handler is responsible for processing + +##### Return + +**Type** + +Schema.SObjectType + +**Description** + +The instance of `SObjectType` + +--- diff --git a/docs/apex/Log-Management/LoggerSObjectMetadata.md b/docs/apex/Log-Management/LoggerSObjectMetadata.md new file mode 100644 index 000000000..e72c88b3d --- /dev/null +++ b/docs/apex/Log-Management/LoggerSObjectMetadata.md @@ -0,0 +1,97 @@ +--- +layout: default +--- + +## LoggerSObjectMetadata class + +Provides details to LWCs about Logger's `SObjects`, using `@AuraEnabled` properties + +--- + +### Methods + +#### `getSchema(Schema.SObjectType sobjectType)` → `SObjectSchema` + +Provides schema details about the specified `SObjectType` + +##### Parameters + +| Param | Description | +| ------------- | ------------------------------------------------------------------------------------------------ | +| `sobjectType` | The instance of `SObjectType` to convert to an instance of `LoggerSObjectMetadata.SObjectSchema` | + +##### Return + +**Type** + +SObjectSchema + +**Description** + +An instance of `LoggerSObjectMetadata.SObjectSchema` for the specified `SObjectType` + +#### `getSchemaForName(String sobjectApiName)` → `SObjectSchema` + +Provides schema details about the specified `SObjectType` + +##### Parameters + +| Param | Description | +| ---------------- | ---------------------------------------------------------------------------------------------------- | +| `sobjectApiName` | The API name of the `SObjectType` to convert to an instance of `LoggerSObjectMetadata.SObjectSchema` | + +##### Return + +**Type** + +SObjectSchema + +**Description** + +An instance of `LoggerSObjectMetadata.SObjectSchema` for the specified `SObjectType` + +--- + +### Inner Classes + +#### LoggerSObjectMetadata.FieldSchema class + +Inner class for `SObjectField` details to LWCs, using `@AuraEnabled` properties + +--- + +##### Properties + +###### `apiName` → `String` + +###### `inlineHelpText` → `String` + +###### `label` → `String` + +###### `localApiName` → `String` + +###### `type` → `String` + +--- + +#### LoggerSObjectMetadata.SObjectSchema class + +Inner class for `SObject` details to LWCs, using `@AuraEnabled` properties + +--- + +##### Properties + +###### `apiName` → `String` + +###### `fields` → `Map` + +###### `label` → `String` + +###### `labelPlural` → `String` + +###### `localApiName` → `String` + +###### `namespacePrefix` → `String` + +--- diff --git a/docs/apex/log-management/LoggerSettingsController.md b/docs/apex/Log-Management/LoggerSettingsController.md similarity index 97% rename from docs/apex/log-management/LoggerSettingsController.md rename to docs/apex/Log-Management/LoggerSettingsController.md index 8c0bb6491..7b5eb09b2 100644 --- a/docs/apex/log-management/LoggerSettingsController.md +++ b/docs/apex/Log-Management/LoggerSettingsController.md @@ -135,6 +135,10 @@ Inner class for returning all custom `List<PicklistOption>` in a single Ap ###### `loggingLevelOptions` → `List` +###### `platformEventStorageLocationOptions` → `List` + +###### `purgeActionOptions` → `List` + ###### `saveMethodOptions` → `List` ###### `setupOwnerTypeOptions` → `List` diff --git a/docs/apex/log-management/LoggerTagHandler.md b/docs/apex/Log-Management/LoggerTagHandler.md similarity index 100% rename from docs/apex/log-management/LoggerTagHandler.md rename to docs/apex/Log-Management/LoggerTagHandler.md diff --git a/docs/apex/Log-Management/LoggerTriggerableContext.md b/docs/apex/Log-Management/LoggerTriggerableContext.md new file mode 100644 index 000000000..1b0732f11 --- /dev/null +++ b/docs/apex/Log-Management/LoggerTriggerableContext.md @@ -0,0 +1,57 @@ +--- +layout: default +--- + +## LoggerTriggerableContext class + +Class used by the logging system for trigger contextual details + +### Related + +[LoggerSObjectHandler](LoggerSObjectHandler) + +[LoggerPlugin](LoggerPlugin) + +--- + +### Constructors + +#### `LoggerTriggerableContext(Schema.SObjectType sobjectType,TriggerOperation triggerOperationType,List triggerNew,Map triggerNewMap,Map triggerOldMap)` + +--- + +### Properties + +#### `sobjectType` → `Schema.SObjectType` + +#### `sobjectTypeName` → `String` + +#### `triggerNew` → `List` + +#### `triggerNewMap` → `Map` + +#### `triggerOldMap` → `Map` + +#### `triggerOperationType` → `TriggerOperation` + +#### `triggerOperationTypeName` → `String` + +#### `triggerRecords` → `List` + +--- + +### Inner Classes + +#### LoggerTriggerableContext.RecordInput class + +Class used by the logging system to provide trigger record details to Flow + +--- + +##### Properties + +###### `triggerRecordNew` → `SObject` + +###### `triggerRecordOld` → `SObject` + +--- diff --git a/docs/apex/log-management/RelatedLogEntriesController.md b/docs/apex/Log-Management/RelatedLogEntriesController.md similarity index 100% rename from docs/apex/log-management/RelatedLogEntriesController.md rename to docs/apex/Log-Management/RelatedLogEntriesController.md diff --git a/docs/apex/logger-engine/ComponentLogger.md b/docs/apex/Logger-Engine/ComponentLogger.md similarity index 100% rename from docs/apex/logger-engine/ComponentLogger.md rename to docs/apex/Logger-Engine/ComponentLogger.md diff --git a/docs/apex/logger-engine/FlowCollectionLogEntry.md b/docs/apex/Logger-Engine/FlowCollectionLogEntry.md similarity index 100% rename from docs/apex/logger-engine/FlowCollectionLogEntry.md rename to docs/apex/Logger-Engine/FlowCollectionLogEntry.md diff --git a/docs/apex/logger-engine/FlowLogEntry.md b/docs/apex/Logger-Engine/FlowLogEntry.md similarity index 100% rename from docs/apex/logger-engine/FlowLogEntry.md rename to docs/apex/Logger-Engine/FlowLogEntry.md diff --git a/docs/apex/logger-engine/FlowLogger.md b/docs/apex/Logger-Engine/FlowLogger.md similarity index 100% rename from docs/apex/logger-engine/FlowLogger.md rename to docs/apex/Logger-Engine/FlowLogger.md diff --git a/docs/apex/logger-engine/FlowRecordLogEntry.md b/docs/apex/Logger-Engine/FlowRecordLogEntry.md similarity index 100% rename from docs/apex/logger-engine/FlowRecordLogEntry.md rename to docs/apex/Logger-Engine/FlowRecordLogEntry.md diff --git a/docs/apex/logger-engine/LogEntryEventBuilder.md b/docs/apex/Logger-Engine/LogEntryEventBuilder.md similarity index 93% rename from docs/apex/logger-engine/LogEntryEventBuilder.md rename to docs/apex/Logger-Engine/LogEntryEventBuilder.md index aea6a5c65..88889d6db 100644 --- a/docs/apex/logger-engine/LogEntryEventBuilder.md +++ b/docs/apex/Logger-Engine/LogEntryEventBuilder.md @@ -335,6 +335,46 @@ LogEntryEventBuilder The same instance of `LogEntryEventBuilder`, useful for chaining methods +#### `setHttpRequestDetails(HttpRequest request)` → `LogEntryEventBuilder` + +Sets the log entry event's HTTP Request fields + +##### Parameters + +| Param | Description | +| --------- | ------------------------------------ | +| `request` | The instance of `HttpRequest` to log | + +##### Return + +**Type** + +LogEntryEventBuilder + +**Description** + +The same instance of `LogEntryEventBuilder`, useful for chaining methods + +#### `setHttpResponseDetails(HttpResponse response)` → `LogEntryEventBuilder` + +Sets the log entry event's HTTP Response fields + +##### Parameters + +| Param | Description | +| ---------- | ------------------------------------- | +| `response` | The instance of `HttpResponse` to log | + +##### Return + +**Type** + +LogEntryEventBuilder + +**Description** + +The same instance of `LogEntryEventBuilder`, useful for chaining methods + #### `setMessage(LogMessage logMessage)` → `LogEntryEventBuilder` Sets the log entry event's message field diff --git a/docs/apex/logger-engine/LogMessage.md b/docs/apex/Logger-Engine/LogMessage.md similarity index 100% rename from docs/apex/logger-engine/LogMessage.md rename to docs/apex/Logger-Engine/LogMessage.md diff --git a/docs/apex/logger-engine/Logger.md b/docs/apex/Logger-Engine/Logger.md similarity index 99% rename from docs/apex/logger-engine/Logger.md rename to docs/apex/Logger-Engine/Logger.md index 68889c89a..a95b3e501 100644 --- a/docs/apex/logger-engine/Logger.md +++ b/docs/apex/Logger-Engine/Logger.md @@ -3216,6 +3216,20 @@ LoggingLevel The matching instance of LoggingLevel (or a default value if a match is not found) +#### `getNamespacePrefix()` → `String` + +Returns the current namespace of Nebula Logger + +##### Return + +**Type** + +String + +**Description** + +The current namespace prefix, or an empty string when no namespace is being used + #### `getParentLogTransactionId()` → `String` Returns the transaction ID value that will be used to relate the current transaction's log to a parent log @@ -3334,6 +3348,20 @@ String A string containing the UUID value. +#### `getVersionNumber()` → `String` + +Returns the current version number of Nebula Logger + +##### Return + +**Type** + +String + +**Description** + +The current version number, in the format `v0.0.0` + #### `info(LogMessage logMessage, Database.DeleteResult deleteResult)` → `LogEntryEventBuilder` Creates a new log entry with logging level == `LoggingLevel.INFO` @@ -4200,6 +4228,16 @@ Saves any entries in Logger's buffer, using the specified save method for o | ------------ | ------------------------------------------------------------------------- | | `saveMethod` | The enum value of Logger.SaveMethod to use for this specific save action. | +#### `saveLog(String saveMethodName)` → `void` + +Saves any entries in Logger's buffer, using the specified save method for only this call. All subsequent calls to saveLog() will use the transaction save method. + +##### Parameters + +| Param | Description | +| ---------------- | ------------------------------------------------------------------------- | +| `saveMethodName` | The String value of the save method to use for this specific save action. | + #### `setParentLogTransactionId(String parentTransactionId)` → `void` Relates the current transaction's log to a parent log via the field Log**c.ParentLog**c This is useful for relating multiple asynchronous operations together, such as batch & queueable jobs. @@ -5008,12 +5046,12 @@ Inner class for publishing log entries via the Queueable interface. ###### `execute(System.QueueableContext queueableContext)` → `void` -Required by the Queueable interface, this method contains the logic executed when the current instance of the queue runs. +Asynchronoulsy publishes the list of `LogEntryEvent__e` records ####### Parameters -| Param | Description | -| ------------------ | --------------------------------- | -| `queueableContext` | The context of the current queue. | +| Param | Description | +| ------------------ | ---------------------------------------------------------- | +| `queueableContext` | The context of the current queue, provided by the platform | --- diff --git a/docs/apex/Logger-Engine/LoggerDataStore.md b/docs/apex/Logger-Engine/LoggerDataStore.md new file mode 100644 index 000000000..02c043111 --- /dev/null +++ b/docs/apex/Logger-Engine/LoggerDataStore.md @@ -0,0 +1,537 @@ +--- +layout: default +--- + +## LoggerDataStore class + +Class used to manage any data-related operations, including database DML statements, publishing platform events via the event bus, and enqueueing queueable jobs + +--- + +### Methods + +#### `getDatabase()` → `Database` + +The instance `LoggerDataStore.Database` used for any DML operations in the current transaction. + +##### Return + +**Type** + +Database + +**Description** + +The singleton instance of `LoggerDataStore.Database` + +#### `getEventBus()` → `EventBus` + +The instance `LoggerDataStore.EventBus` used for publishing platform events in the current transaction. + +##### Return + +**Type** + +EventBus + +**Description** + +The singleton instance of `LoggerDataStore.EventBus` + +#### `getJobQueue()` → `JobQueue` + +The instance `LoggerDataStore.JobQueue` used for enqueuing any queueable jobs in the current transaction. + +##### Return + +**Type** + +JobQueue + +**Description** + +The singleton instance of `LoggerDataStore.JobQueue` + +--- + +### Inner Classes + +#### LoggerDataStore.Database class + +Class used to centralize the handling of any DML operations + +--- + +##### Methods + +###### `deleteRecord(SObject record)` → `Database.DeleteResult` + +Executes a `delete` DML operation for the `SObject` record + +####### Parameters + +| Param | Description | +| -------- | ------------------------------ | +| `record` | The `SObject` record to delete | + +####### Return + +**Type** + +Database.DeleteResult + +**Description** + +The instance of `Database.DeleteResult`, generated by the platform when deleting the record + +###### `deleteRecords(List records)` → `List` + +Executes a `delete` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| --------- | --------------------------------------- | +| `records` | The list of `SObject` records to delete | + +####### Return + +**Type** + +List<Database.DeleteResult> + +**Description** + +The instance of `List<Database.DeleteResult>`, generated by the platform when deleting the records + +###### `deleteRecords(List records, Boolean allOrNone)` → `List` + +Executes a `delete` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ----------- | ------------------------------------------------------------------------------------------------- | +| `records` | The list of `SObject` records to delete | +| `allOrNone` | Controls if all records must be deleted (`true`), or if partial deletion should be used (`false`) | + +####### Return + +**Type** + +List<Database.DeleteResult> + +**Description** + +The instance of `List<Database.DeleteResult>`, generated by the platform when deleting the records + +###### `hardDeleteRecord(SObject record)` → `Database.DeleteResult` + +Executes a `delete` DML operation for the `SObject` record, followed by hard deleting the record using `Database.emptyRecycleBin(record)` + +####### Parameters + +| Param | Description | +| -------- | ------------------------------ | +| `record` | The `SObject` record to delete | + +####### Return + +**Type** + +Database.DeleteResult + +**Description** + +The instance of `Database.DeleteResult`, generated by the platform when deleting the record + +###### `hardDeleteRecords(List records)` → `List` + +Executes a `delete` DML operation on the provided list of `SObject` records, followed by hard deleting the records using `Database.emptyRecycleBin(records)` + +####### Parameters + +| Param | Description | +| --------- | --------------------------------------- | +| `records` | The list of `SObject` records to delete | + +####### Return + +**Type** + +List<Database.DeleteResult> + +**Description** + +The instance of `List<Database.DeleteResult>`, generated by the platform when deleting the records + +###### `insertRecord(SObject record)` → `Database.SaveResult` + +Executes an `insert` DML operation on the provided `SObject` record + +####### Parameters + +| Param | Description | +| -------- | ------------------------------ | +| `record` | The `SObject` record to insert | + +####### Return + +**Type** + +Database.SaveResult + +**Description** + +The instance of `Database.SaveResult`, generated by the platform when creating the record + +###### `insertRecords(List records)` → `List` + +Executes an `insert` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| --------- | --------------------------------------- | +| `records` | The list of `SObject` records to insert | + +####### Return + +**Type** + +List<Database.SaveResult> + +**Description** + +The instance of `List<Database.SaveResult>`, generated by the platform when creating the records + +###### `insertRecords(List records, Boolean allOrNone)` → `List` + +Executes an `insert` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ----------- | ------------------------------------------------------------------------------------------------- | +| `records` | The list of `SObject` records to insert | +| `allOrNone` | Controls if all records must be created (`true`), or if partial creation should be used (`false`) | + +####### Return + +**Type** + +List<Database.SaveResult> + +**Description** + +The instance of `List<Database.SaveResult>`, generated by the platform when creating the records + +###### `insertRecords(List records, Database.DmlOptions dmlOptions)` → `List` + +Executes an `insert` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ------------ | ------------------------------------------------------------------------------------- | +| `records` | The list of `SObject` records to insert | +| `dmlOptions` | Controls additional DML options, using the provided instance of `Database.DmlOptions` | + +####### Return + +**Type** + +List<Database.SaveResult> + +**Description** + +The instance of `List<Database.SaveResult>`, generated by the platform when creating the records + +###### `undeleteRecord(SObject record)` → `Database.UndeleteResult` + +Executes an `undelete` DML operation on the provided `SObject` record + +####### Parameters + +| Param | Description | +| -------- | -------------------------------- | +| `record` | The `SObject` record to undelete | + +####### Return + +**Type** + +Database.UndeleteResult + +**Description** + +The instance of `Database.UndeleteResult`, generated by the platform when undeleting the record + +###### `undeleteRecords(List records)` → `List` + +Executes an `undelete` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| --------- | ----------------------------------------- | +| `records` | The list of `SObject` records to undelete | + +####### Return + +**Type** + +List<Database.UndeleteResult> + +**Description** + +The instance of `List<Database.UndeleteResult>`, generated by the platform when deleting the records + +###### `undeleteRecords(List records, Boolean allOrNone)` → `List` + +Executes an `undelete` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ----------- | ----------------------------------------------------------------------------------------------------- | +| `records` | The list of `SObject` records to undelete | +| `allOrNone` | Controls if all records must be undeleted (`true`), or if partial undeletion should be used (`false`) | + +####### Return + +**Type** + +List<Database.UndeleteResult> + +**Description** + +The instance of `List<Database.DeleteResult>`, generated by the platform when undeleting the records + +###### `updateRecord(SObject record)` → `Database.SaveResult` + +Executes an `update` DML operation on the provided `SObject` record + +####### Parameters + +| Param | Description | +| -------- | ------------------------------ | +| `record` | The `SObject` record to update | + +####### Return + +**Type** + +Database.SaveResult + +**Description** + +The instance of `Database.SaveResult`, generated by the platform when updating the record + +###### `updateRecords(List records)` → `List` + +Executes an `update` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| --------- | --------------------------------------- | +| `records` | The list of `SObject` records to update | + +####### Return + +**Type** + +List<Database.SaveResult> + +**Description** + +The instance of `List<Database.SaveResult>`, generated by the platform when updating the records + +###### `updateRecords(List records, Boolean allOrNone)` → `List` + +Executes an `update` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ----------- | ------------------------------------------------------------------------------------------------ | +| `records` | The list of `SObject` records to update | +| `allOrNone` | Controls if all records must be updated (`true`), or if partial updates should be used (`false`) | + +####### Return + +**Type** + +List<Database.SaveResult> + +**Description** + +The instance of `List<Database.SaveResult>`, generated by the platform when updating the records + +###### `updateRecords(List records, Database.DmlOptions dmlOptions)` → `List` + +Executes an `update` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ------------ | ------------------------------------------------------------------------------------- | +| `records` | The list of `SObject` records to update | +| `dmlOptions` | Controls additional DML options, using the provided instance of `Database.DmlOptions` | + +####### Return + +**Type** + +List<Database.SaveResult> + +**Description** + +The instance of `List<Database.SaveResult>`, generated by the platform when updating the records + +###### `upsertRecord(SObject record, Schema.SObjectField externalIdField)` → `Database.UpsertResult` + +Executes an `upsert` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ----------------- | ---------------------------------------------------------------------------------------- | +| `record` | The `SObject` record to update | +| `externalIdField` | The `SObjectField` of the external ID field on the target `SObject` to use for upserting | + +####### Return + +**Type** + +Database.UpsertResult + +**Description** + +The instance of `Database.UpsertResult`, generated by the platform when upserting the record + +###### `upsertRecords(List records, Schema.SObjectField externalIdField)` → `List` + +Executes an `upsert` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ----------------- | ---------------------------------------------------------------------------------------- | +| `records` | The list of `SObject` records to upsert | +| `externalIdField` | The `SObjectField` of the external ID field on the target `SObject` to use for upserting | + +####### Return + +**Type** + +List<Database.UpsertResult> + +**Description** + +The instance of `List<Database.UpsertResult>`, generated by the platform when Upserting the records + +###### `upsertRecords(List records, Schema.SObjectField externalIdField, Boolean allOrNone)` → `List` + +Executes an `upsert` DML operation on the provided list of `SObject` records + +####### Parameters + +| Param | Description | +| ----------------- | ------------------------------------------------------------------------------------------------ | +| `records` | The list of `SObject` records to upsert | +| `externalIdField` | The `SObjectField` of the external ID field on the target `SObject` to use for upserting | +| `allOrNone` | Controls if all records must be updated (`true`), or if partial updates should be used (`false`) | + +####### Return + +**Type** + +List<Database.UpsertResult> + +**Description** + +The instance of `List<Database.UpsertResult>`, generated by the platform when Upserting the records + +--- + +#### LoggerDataStore.EventBus class + +Class used to centralize the handling of any platform event publishing operations + +--- + +##### Methods + +###### `publishRecord(SObject platformEvent)` → `Database.SaveResult` + +Publishes a single platform event record, using `EventBus.publish(SObject record); + +####### Parameters + +| Param | Description | +| --------------- | ------------------------------------ | +| `platformEvent` | The platform event record to publish | + +####### Return + +**Type** + +Database.SaveResult + +**Description** + +The instance of `Database.SaveResult`, generated by the platform when publishing the platform event record + +###### `publishRecords(List platformEvents)` → `List` + +Publishes a list of platform event records, using `EventBus.publish(List<SObject> records); + +####### Parameters + +| Param | Description | +| ---------------- | --------------------------------------------- | +| `platformEvents` | The list of platform event records to publish | + +####### Return + +**Type** + +List<Database.SaveResult> + +**Description** + +The instance of `List<Database.SaveResult>`, generated by the platform when publishing the platform event records + +--- + +#### LoggerDataStore.JobQueue class + +Class used to centralize the handling of enqueueing any queueable jobs + +--- + +##### Methods + +###### `enqueueJob(Queueable queueableJob)` → `Id` + +Enqueues a queueable job to execute asynchronously, using `System.enqueueJob(Queueable queueableJob)` + +####### Parameters + +| Param | Description | +| -------------- | ---------------------------------------------------------- | +| `queueableJob` | An instance of a `Queueable` class that should be enqueued | + +####### Return + +**Type** + +Id + +**Description** + +The `Id` of the queueable job + +--- diff --git a/docs/apex/Logger-Engine/LoggerSObjectHandler.md b/docs/apex/Logger-Engine/LoggerSObjectHandler.md new file mode 100644 index 000000000..0264de571 --- /dev/null +++ b/docs/apex/Logger-Engine/LoggerSObjectHandler.md @@ -0,0 +1,98 @@ +--- +layout: default +--- + +## LoggerSObjectHandler class + +Abstract class used by trigger handlers for shared logic + +--- + +### Constructors + +#### `LoggerSObjectHandler()` + +## Default constructor + +### Methods + +#### `execute()` → `void` + +Runs the handler class's logic, as well as any configured plugins + +#### `getHandler(Schema.SObjectType sobjectType)` → `LoggerSObjectHandler` + +Returns an instance of `LoggerSObjectHandler` that has been built & configured for the specified `SObjectType` + +##### Parameters + +| Param | Description | +| ------------- | --------------------------------------------------------------------------------------- | +| `sobjectType` | The instance `SObjectType` to check for a configured instance of `LoggerSObjectHandler` | + +##### Return + +**Type** + +LoggerSObjectHandler + +**Description** + +The Apex class that extends `LoggerSObjectHandler` and has been configured for the specified `SObjectType` + +#### `getHandler(Schema.SObjectType sobjectType, LoggerSObjectHandler defaultImplementation)` → `LoggerSObjectHandler` + +Returns an instance of `LoggerSObjectHandler` that has been built & configured for the specified `SObjectType` + +##### Parameters + +| Param | Description | +| ----------------------- | ----------------------------------------------------------------------------------------------------------------------------------- | +| `sobjectType` | The instance `SObjectType` to check for a configured instance of `LoggerSObjectHandler` | +| `defaultImplementation` | A default implementation of `LoggerSObjectHandler` that should be used, if a configuration cannot be found `LoggerSObjectHandler_t` | + +##### Return + +**Type** + +LoggerSObjectHandler + +**Description** + +The Apex class that extends `LoggerSObjectHandler` and has been configured for the specified `SObjectType` + +#### `getSObjectType()` → `Schema.SObjectType` + +Returns the SObject Type that the handler is responsible for processing + +##### Return + +**Type** + +Schema.SObjectType + +**Description** + +The instance of `SObjectType` + +#### `overrideTriggerableContext(LoggerTriggerableContext input)` → `LoggerSObjectHandler` + +Provides the ability to override the instance `LoggerTriggerableContext` that is normally provided internally by `LoggerSObjectHandler` + +##### Parameters + +| Param | Description | +| ------- | ---------------------------------------------------------------------------------- | +| `input` | The instance of `LoggerTriggerableContext` to use within the trigger handler class | + +##### Return + +**Type** + +LoggerSObjectHandler + +**Description** + +The same instance of `LoggerSObjectHandler`, useful for chaining methods + +--- diff --git a/docs/apex/Logger-Engine/LoggerTriggerableContext.md b/docs/apex/Logger-Engine/LoggerTriggerableContext.md new file mode 100644 index 000000000..7662ab991 --- /dev/null +++ b/docs/apex/Logger-Engine/LoggerTriggerableContext.md @@ -0,0 +1,59 @@ +--- +layout: default +--- + +## LoggerTriggerableContext class + +Class used by the logging system for trigger contextual details + +### Related + +[LoggerSObjectHandler](LoggerSObjectHandler) + +[LoggerPlugin](LoggerPlugin) + +--- + +### Constructors + +#### `LoggerTriggerableContext(Schema.SObjectType sobjectType, TriggerOperation triggerOperationType, List triggerNew)` + +#### `LoggerTriggerableContext(Schema.SObjectType sobjectType,TriggerOperation triggerOperationType,List triggerNew,Map triggerNewMap,Map triggerOldMap)` + +--- + +### Properties + +#### `sobjectType` → `Schema.SObjectType` + +#### `sobjectTypeName` → `String` + +#### `triggerNew` → `List` + +#### `triggerNewMap` → `Map` + +#### `triggerOldMap` → `Map` + +#### `triggerOperationType` → `TriggerOperation` + +#### `triggerOperationTypeName` → `String` + +#### `triggerRecords` → `List` + +--- + +### Inner Classes + +#### LoggerTriggerableContext.RecordInput class + +Class used by the logging system to provide trigger record details to Flow + +--- + +##### Properties + +###### `triggerRecordNew` → `SObject` + +###### `triggerRecordOld` → `SObject` + +--- diff --git a/docs/apex/Plugins/LogEntryArchiveBuilder.md b/docs/apex/Plugins/LogEntryArchiveBuilder.md new file mode 100644 index 000000000..7b3b9a0a4 --- /dev/null +++ b/docs/apex/Plugins/LogEntryArchiveBuilder.md @@ -0,0 +1,57 @@ +--- +layout: default +--- + +## LogEntryArchiveBuilder class + +Builder class to create an instance of `LogEntryArchive__b`, used by the BigObject plugin + +### Related + +[LogEntryArchivePlugin](../Plugins/LogEntryArchivePlugin) + +[LogEntryEventBuilder](LogEntryEventBuilder) + +--- + +### Constructors + +#### `LogEntryArchiveBuilder(LogEntryEvent__e logEntryEvent)` + +Used by `LogEntryArchivePlugin` to instantiate a new instance of `LogEntryArchiveBuilder` + +##### Parameters + +| Param | Description | +| --------------- | ------------------------------------------------------------------------------------- | +| `logEntryEvent` | The `LogEntryEvent__e` record that will be converted to a `LogEntryArchive__b` record | + +#### `LogEntryArchiveBuilder(LogEntry__c logEntry)` + +Used by `LogEntryArchivePlugin` to instantiate a new instance of `LogEntryArchiveBuilder` + +##### Parameters + +| Param | Description | +| ---------- | -------------------------------------------------------------------------------- | +| `logEntry` | The `LogEntry__c` record that will be converted to a `LogEntryArchive__b` record | + +--- + +### Methods + +#### `getLogEntryArchive()` → `LogEntryArchive__b` + +Returns the `LogEntryArchive__b` record for this instance of LogEntryEventBuilder + +##### Return + +**Type** + +LogEntryArchive\_\_b + +**Description** + +The `LogEntryArchive__b` record + +--- diff --git a/docs/apex/Plugins/LogEntryArchiveController.md b/docs/apex/Plugins/LogEntryArchiveController.md new file mode 100644 index 000000000..1a9123684 --- /dev/null +++ b/docs/apex/Plugins/LogEntryArchiveController.md @@ -0,0 +1,33 @@ +--- +layout: default +--- + +## LogEntryArchiveController class + +Controller class used by the LWC `logEntryArchives` to display `LogEntryArchive__b` data + +### Related + +[LogEntryArchivePlugin](../Plugins/LogEntryArchivePlugin) + +[LogEntryEventBuilder](LogEntryEventBuilder) + +--- + +### Methods + +#### `getLogEntryArchives()` → `List` + +Returns a list of `LogEntryArchive__b` records, based on the current user's record access + +##### Return + +**Type** + +List<LogEntryArchive\_\_b> + +**Description** + +The list of `LogEntryArchive__b` records + +--- diff --git a/docs/apex/Plugins/LogEntryArchivePlugin.md b/docs/apex/Plugins/LogEntryArchivePlugin.md new file mode 100644 index 000000000..095b6a696 --- /dev/null +++ b/docs/apex/Plugins/LogEntryArchivePlugin.md @@ -0,0 +1,68 @@ +--- +layout: default +--- + +## LogEntryArchivePlugin class + +Optional plugin that provides a BigObject, `LogEntryArchive__b`, as an alternative option to the platform event `LogEntryEvent__e` + +### Related + +[LoggerPlugin](LoggerPlugin) + +--- + +### Constructors + +#### `LogEntryArchivePlugin()` + +## Default constructor + +### Methods + +#### `execute(LoggerPlugin_t configuration, LoggerBatchableContext input, List loggerRecords)` → `void` + +Converts any `LogEntry__c` records into `LogEntryArchive__b` records + +##### Parameters + +| Param | Description | +| --------------- | ------------------------------------------------------------------------ | +| `configuration` | The instance of `LoggerPlugin_t` configured for this specific plugin | +| `input` | The instance of `LoggerBatchableContext`, provided by the logging system | +| `loggerRecords` | The list of `SObject` scope records provider by `LogBatchPurger` | + +#### `execute(LoggerPlugin_t configuration, LoggerTriggerableContext input)` → `void` + +Handles converting Logger's buffer of `LogEntryEvent__e` records into `LogEntryArchive__b` records for any user with the included custom save method 'BIG_OBJECT' + +##### Parameters + +| Param | Description | +| --------------- | -------------------------------------------------------------------------- | +| `configuration` | The instance of `LoggerPlugin_t` configured for this specific plugin | +| `input` | The instance of `LoggerTriggerableContext`, provided by the logging system | + +#### `finish(LoggerPlugin_t configuration, LoggerBatchableContext input)` → `void` + +No-op method, required by the interface `LoggerPlugin.Batchable` + +##### Parameters + +| Param | Description | +| --------------- | ------------------------------------------------------------------------ | +| `configuration` | The instance of `LoggerPlugin_t` configured for this specific plugin | +| `input` | The instance of `LoggerBatchableContext`, provided by the logging system | + +#### `start(LoggerPlugin_t configuration, LoggerBatchableContext input)` → `void` + +Skips directly deleting `LogEntryTag__c` records in `LogBatchPurger` so that the tags can be included when `LogEntry__c` records are archived into `LogEntryArchive__b` + +##### Parameters + +| Param | Description | +| --------------- | ------------------------------------------------------------------------ | +| `configuration` | The instance of `LoggerPlugin_t` configured for this specific plugin | +| `input` | The instance of `LoggerBatchableContext`, provided by the logging system | + +--- diff --git a/docs/apex/Plugins/LogRetentionRulesPlugin.md b/docs/apex/Plugins/LogRetentionRulesPlugin.md new file mode 100644 index 000000000..d616a94fc --- /dev/null +++ b/docs/apex/Plugins/LogRetentionRulesPlugin.md @@ -0,0 +1,54 @@ +--- +layout: default +--- + +## LogRetentionRulesPlugin class + +Optional plugin that adds the ability to create & deploy advanced, configurable rules for setting the retention date of `Log__c` records, using custom metadata types `LogRetentionRule_t` and `LogRetentionRuleCondition_t`. + +--- + +### Properties + +#### `conditions` → `List` + +#### `conditionsLogic` → `String` + +#### `conditionsLogicType` → `String` + +#### `matchesFilter` → `Boolean` + +#### `record` → `SObject` + +#### `rule` → `LogRetentionRule_t` + +--- + +### Methods + +#### `FieldPath(Schema.SObjectType sobjectType, String fieldPath)` → `public` + +#### `FilterResult(SObject record, LogRetentionRule_t rule, List filterConditions)` → `public` + +#### `evaluate(String x)` → `Boolean` + +#### `execute(LoggerPlugin_t configuration, LoggerTriggerableContext input)` → `void` + +Handles converting Logger's buffer of `LogEntryEvent__e` records into `LogEntryArchive__b` records for any user with the included custom save method 'BIG_OBJECT' + +##### Parameters + +| Param | Description | +| --------------- | -------------------------------------------------------------------------- | +| `configuration` | The instance of `LoggerPlugin_t` configured for this specific plugin | +| `input` | The instance of `LoggerTriggerableContext`, provided by the logging system | + +#### `getCondition()` → `String` + +#### `getField()` → `Schema.SObjectField` + +#### `getValue(SObject record)` → `Object` + +#### `matchesFilter()` → `Boolean` + +--- diff --git a/docs/apex/Plugins/SlackLoggerPlugin.md b/docs/apex/Plugins/SlackLoggerPlugin.md new file mode 100644 index 000000000..295878e05 --- /dev/null +++ b/docs/apex/Plugins/SlackLoggerPlugin.md @@ -0,0 +1,82 @@ +--- +layout: default +--- + +## SlackLoggerPlugin class + +Optional plugin that integrates with Slack to send alerts for important logs + +--- + +### Constructors + +#### `SlackLoggerPlugin()` + +## Default constructor + +### Properties + +#### `actions` → `List` + +#### `attachments` → `List` + +#### `author_icon` → `String` + +#### `author_link` → `String` + +#### `author_name` → `String` + +#### `color` → `String` + +#### `fallback` → `String` + +#### `fields` → `List` + +#### `isShort` → `Boolean` + +#### `pretext` → `String` + +#### `text` → `String` + +#### `text` → `String` + +#### `text` → `String` + +#### `title` → `String` + +#### `title` → `String` + +#### `title_link` → `String` + +#### `type` → `String` + +#### `url` → `String` + +#### `value` → `String` + +--- + +### Methods + +#### `execute(LoggerPlugin_t configuration, LoggerTriggerableContext input)` → `void` + +Handles the integration with Slack. This method is automatically called by Nebula Logger's plugin framework. + +##### Parameters + +| Param | Description | +| --------------- | -------------------------------------------------------------------------- | +| `configuration` | The instance of `LoggerPlugin_t` configured for this specific plugin | +| `input` | The instance of `LoggerTriggerableContext`, provided by the logging system | + +#### `execute(System.QueueableContext queueableContext)` → `void` + +Handles the queuable execute logic. Required by the Queueable interface. + +##### Parameters + +| Param | Description | +| ------------------ | ----------------------------------------- | +| `queueableContext` | Context of the current queuable instance. | + +--- diff --git a/docs/apex/Test-Utilities/LoggerMockDataCreator.md b/docs/apex/Test-Utilities/LoggerMockDataCreator.md new file mode 100644 index 000000000..3c1ede437 --- /dev/null +++ b/docs/apex/Test-Utilities/LoggerMockDataCreator.md @@ -0,0 +1,683 @@ +--- +layout: default +--- + +## LoggerMockDataCreator class + +Utility class used to help with generating mock data when writing Apex tests for Nebula Logger. These methods are generic, and should work in any Salesforce org. These methods can be used when writing Apex tests for plugins. + +### Related + +[LoggerMockDataStore](../Test-Utilities/LoggerMockDataStore) + +[LoggerTestConfigurator](../Test-Utilities/LoggerTestConfigurator) + +--- + +### Methods + +#### `createAggregateResult()` → `AggregateResult` + +Instances of `AggregateResult` can not be created directly in Apex. This method uses a workaround to generate a mock. + +##### Return + +**Type** + +AggregateResult + +**Description** + +The mock instance of `AggregateResult` + +#### `createAggregateResult(Map mockAggregateKeyValues)` → `AggregateResult` + +Instances of `AggregateResult` can not be created directly in Apex. This method uses a workaround to generate a mock, using the provided map of aliases & aggregate values + +##### Parameters + +| Param | Description | +| ------------------------ | --------------------------------------------------------------------------------------- | +| `mockAggregateKeyValues` | A map of aliases & aggregate values to use when creating the mock `AggregateResult` | + +##### Return + +**Type** + +AggregateResult + +**Description** + +The mock instance of `AggregateResult` + +#### `createBatchableContext(String jobId)` → `MockBatchableContext` + +Creates an instance of the class `MockBatchableContext` that implements the interface `Database.BatchableContext`. This can be used when testing batch jobs. + +##### Parameters + +| Param | Description | +| ------- | --------------------------------------------------------------------------------------- | +| `jobId` | A string value to use as the batchable job ID - this can be a true ID, or just a string | + +##### Return + +**Type** + +MockBatchableContext + +**Description** + +The instance of `MockBatchableContext` + +#### `createDataBuilder(Schema.SObjectType sobjectType)` → `SObjectTestDataBuilder` + +Creates a new builder instance for the specified `SObjectType`, including creating a new `SObject` record. The new `SObject` record is created with any default field values that have been configured on the `SObjectType`. + +##### Parameters + +| Param | Description | +| ------------- | ------------------------------------------------------------------- | +| `sobjectType` | The `SObjectType` to use for generating a new test `SObject` record | + +##### Return + +**Type** + +SObjectTestDataBuilder + +**Description** + +A new instance of `SObjectTestDataBuilder` for the specified `SObjectType` + +#### `createDataBuilder(SObject record)` → `SObjectTestDataBuilder` + +Creates a new builder instance for the specified `SObject` record + +##### Parameters + +| Param | Description | +| -------- | --------------------------------------------------------------- | +| `record` | The existing test `SObject` record to populate with sample data | + +##### Return + +**Type** + +SObjectTestDataBuilder + +**Description** + +A new instance of `SObjectTestDataBuilder` for the specified `SObject` + +#### `createDatabaseDeleteResult(Boolean isSuccess)` → `Database.DeleteResult` + +Creates a mock instance of `Database.DeleteResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests). A fake record ID is automatically included. + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | + +##### Return + +**Type** + +Database.DeleteResult + +**Description** + +The mock instance of `Database.DeleteResult` + +#### `createDatabaseDeleteResult(Boolean isSuccess, Id recordId)` → `Database.DeleteResult` + +Creates a mock instance of `Database.DeleteResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests) + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | +| `recordId` | The record ID to use within the mock result | + +##### Return + +**Type** + +Database.DeleteResult + +**Description** + +The mock instance of `Database.DeleteResult` + +#### `createDatabaseMergeResult(Boolean isSuccess)` → `Database.MergeResult` + +Creates a mock instance of `Database.MergeResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests). A fake record ID is automatically included. + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | + +##### Return + +**Type** + +Database.MergeResult + +**Description** + +The mock instance of `Database.MergeResult` + +#### `createDatabaseMergeResult(Boolean isSuccess, Id recordId)` → `Database.MergeResult` + +Creates a mock instance of `Database.MergeResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests) + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | +| `recordId` | The record ID to use within the mock result | + +##### Return + +**Type** + +Database.MergeResult + +**Description** + +The mock instance of `Database.MergeResult` + +#### `createDatabaseSaveResult(Boolean isSuccess)` → `Database.SaveResult` + +Creates a mock instance of `Database.SaveResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests). A fake record ID is automatically included. + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | + +##### Return + +**Type** + +Database.SaveResult + +**Description** + +The mock instance of `Database.SaveResult` + +#### `createDatabaseSaveResult(Boolean isSuccess, Id recordId)` → `Database.SaveResult` + +Creates a mock instance of `Database.SaveResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests) + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | +| `recordId` | The record ID to use within the mock result | + +##### Return + +**Type** + +Database.SaveResult + +**Description** + +The mock instance of `Database.SaveResult` + +#### `createDatabaseUndeleteResult(Boolean isSuccess)` → `Database.UndeleteResult` + +Creates a mock instance of `Database.UndeleteResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests). A fake record ID is automatically included. + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | + +##### Return + +**Type** + +Database.UndeleteResult + +**Description** + +The mock instance of `Database.UndeleteResult` + +#### `createDatabaseUndeleteResult(Boolean isSuccess, Id recordId)` → `Database.UndeleteResult` + +Creates a mock instance of `Database.UndeleteResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests) + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | +| `recordId` | The record ID to use within the mock result | + +##### Return + +**Type** + +Database.UndeleteResult + +**Description** + +The mock instance of `Database.UndeleteResult` + +#### `createDatabaseUpsertResult(Boolean isSuccess, Boolean isCreated)` → `Database.UpsertResult` + +Creates a mock instance of `Database.UpsertResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests). A fake record ID is automatically included. + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | +| `isCreated` | Indicates if the generated mock should have `isCreated` == true | + +##### Return + +**Type** + +Database.UpsertResult + +**Description** + +The mock instance of `Database.UpsertResult` + +#### `createDatabaseUpsertResult(Boolean isSuccess, Boolean isCreated, Id recordId)` → `Database.UpsertResult` + +Creates a mock instance of `Database.UpsertResult` - a mock is used instead of an actual instance to help speed up tests, and to support writing unit tests (instead of integration tests) + +##### Parameters + +| Param | Description | +| ----------- | --------------------------------------------------------------- | +| `isSuccess` | Indicates if the generated mock should have `isSuccess` == true | +| `isCreated` | Indicates if the generated mock should have `isCreated` == true | +| `recordId` | The record ID to use within the mock result | + +##### Return + +**Type** + +Database.UpsertResult + +**Description** + +The mock instance of `Database.UpsertResult` + +#### `createHttpCallout()` → `MockHttpCallout` + +Generates an instance of the class `MockHttpCallout` that implements the interface `HttpCalloutMock`. This can be used when testing batch jobs. + +##### Return + +**Type** + +MockHttpCallout + +**Description** + +The instance of `MockHttpCallout` + +#### `createHttpRequest()` → `HttpRequest` + +Generates an instance of `HttpRequest`. This can be used when testing logging capabilities for instances of `HttpRequest`. + +##### Return + +**Type** + +HttpRequest + +**Description** + +The instance of `HttpRequest` + +#### `createHttpResponse()` → `HttpResponse` + +Generates an instance of `HttpResponse`. This can be used when testing logging capabilities for instances of `HttpResponse`. + +##### Return + +**Type** + +HttpResponse + +**Description** + +The instance of `HttpResponse` + +#### `createId(Schema.SObjectType sobjectType)` → `String` + +Generates a mock record ID for the provided SObject Type + +##### Parameters + +| Param | Description | +| ------------- | ------------------------------------------------- | +| `sobjectType` | The SObject Type for the generated mock record ID | + +##### Return + +**Type** + +String + +**Description** + +The mock record ID for the specified SObject Type + +#### `createUser()` → `User` + +Creates a `User` record for testing purposes, using the current user's profile + +##### Return + +**Type** + +User + +**Description** + +The generated `User` record - it is not automatically inserted into the database. + +#### `createUser(Id profileId)` → `User` + +Creates a `User` record for testing purposes, using the specified profile ID + +##### Parameters + +| Param | Description | +| ----------- | ---------------------------------------------- | +| `profileId` | The `Profile` ID to use for the created `User` | + +##### Return + +**Type** + +User + +**Description** + +The generated `User` record - it is not automatically inserted into the database. + +#### `getNetwork()` → `SObject` + +Returns the current user's `Network` (Experience Cloud site) + +##### Return + +**Type** + +SObject + +**Description** + +The matching `Network` record + +#### `getOrganization()` → `Organization` + +Queries for the `Organization` record for the current environment. + +##### Return + +**Type** + +Organization + +**Description** + +The matching `Organization` record + +#### `getOrganizationEnvironmentType()` → `String` + +Returns the current environment's type - Scratch Org, Sandbox, or Production. + +##### Return + +**Type** + +String + +**Description** + +The environment type + +#### `getUser()` → `User` + +Returns the current user + +##### Return + +**Type** + +User + +**Description** + +The matching `User` record + +#### `getUser(Id userId)` → `User` + +Returns the specified user + +##### Parameters + +| Param | Description | +| -------- | ------------------------------------ | +| `userId` | The ID of the `User` record to query | + +##### Return + +**Type** + +User + +**Description** + +The matching `User` record + +#### `insertQueue(String queueDeveloperName, Schema.SObjectType sobjectType)` → `Group` + +Creates and inserts a `Group` record for testing queues, using the specified SObject Type + +##### Parameters + +| Param | Description | +| -------------------- | --------------------------------------------------------------------------------------------- | +| `queueDeveloperName` | The developer name to use for the new queue (stored in `Group.DeveloperName`) | +| `sobjectType` | The `SObjectType` that the queue should be able to own (stored in `QueueSObject.SObjectType`) | + +##### Return + +**Type** + +Group + +**Description** + +The inserted `Group` record - it is automatically inserted into the database, as well as 1 child `QueueSObject` record. + +#### `setReadOnlyField(SObject record, Schema.SObjectField field, Object value)` → `SObject` + +Sets a value for read-only fields that typically cannot be directly set on some SObjects + +##### Parameters + +| Param | Description | +| -------- | ----------------------------------------------------------- | +| `record` | The `SObject` record to update | +| `field` | The `Schema.SObjectField` for the field to update | +| `value` | The field value to populate on the provied `SObject` record | + +##### Return + +**Type** + +SObject + +**Description** + +A new copy of the original `SObject` record that has the specified read-only field populated + +#### `setReadOnlyField(SObject record, Map changesToFields)` → `SObject` + +Sets values for read-only fields that typically cannot be directly set on some SObjects + +##### Parameters + +| Param | Description | +| ----------------- | -------------------------------------------------------------------------------------------------------- | +| `record` | record description | +| `changesToFields` | An instance of `Map<Schema.SObjectField, Object> containing the read-only fields and corresponding | + +##### Return + +**Type** + +SObject + +**Description** + +A new copy of the original `SObject` record that has the specified read-only fields populated + +--- + +### Inner Classes + +#### LoggerMockDataCreator.MockBatchableContext class + +--- + +##### Constructors + +###### `MockBatchableContext(String jobId)` + +###### `MockBatchableContext(String jobId, Id childJobId)` + +--- + +##### Methods + +###### `getChildJobId()` → `String` + +###### `getJobId()` → `String` + +--- + +#### LoggerMockDataCreator.MockHttpCallout class + +--- + +##### Constructors + +###### `MockHttpCallout()` + +--- + +##### Properties + +###### `request` → `HttpRequest` + +###### `response` → `HttpResponse` + +###### `responseBody` → `String` + +###### `statusCode` → `Integer` + +--- + +##### Methods + +###### `respond(HttpRequest request)` → `HttpResponse` + +###### `setResponseBody(String responseBody)` → `MockHttpCallout` + +###### `setStatusCode(Integer statusCode)` → `MockHttpCallout` + +--- + +#### LoggerMockDataCreator.SObjectTestDataBuilder class + +Class used to create or update an `SObject` record with static fake data. This is useful in situations where you need to have fields populated, but the specific values used are not relevant to a particular test. This class can be used when Apex writing tests for plugins. + +--- + +##### Methods + +###### `getRecord()` → `SObject` + +Returns the builder's `SObject` record with fields populated based on which builder methods have been called + +####### Return + +**Type** + +SObject + +**Description** + +The builder's `SObject` record that was either provided by the calling code, or generated + +###### `populateAllFields()` → `SObjectTestDataBuilder` + +Sets a value on all editable fields, unless the `SObject` record already had a value specified for a field (including `null`) + +####### Return + +**Type** + +SObjectTestDataBuilder + +**Description** + +The `SObject` record, with all editable fields populated + +###### `populateMockId()` → `SObjectTestDataBuilder` + +Generates a mock record ID for the builder's `SObject` record + +####### Parameters + +| Param | Description | +| ------------- | ------------------------------------------------------------------- | +| `sobjectType` | The `SObjectType` to use for generating a new test `SObject` record | +| `record` | The existing test `SObject` record to populate with sample data | + +####### Return + +**Type** + +SObjectTestDataBuilder + +**Description** + +The same instance of `SObjectTestDataBuilder`, useful for chaining methods + +###### `populateRequiredFields()` → `SObjectTestDataBuilder` + +Sets a value on all editable required fields, unless the `SObject` record already had a value specified for a field (including `null`) + +####### Return + +**Type** + +SObjectTestDataBuilder + +**Description** + +The `SObject` record, with all editable required fields populated + +--- diff --git a/docs/apex/Test-Utilities/LoggerMockDataStore.md b/docs/apex/Test-Utilities/LoggerMockDataStore.md new file mode 100644 index 000000000..ceca3c5de --- /dev/null +++ b/docs/apex/Test-Utilities/LoggerMockDataStore.md @@ -0,0 +1,75 @@ +--- +layout: default +--- + +## LoggerMockDataStore class + +Utility class used to mock any data-related operations for the database, event bus, and queueable jobs. These methods are generic, and should work in any Salesforce org. These methods can be used when writing Apex tests for plugins. + +### Related + +[LoggerDataStore](LoggerDataStore) + +[LoggerMockDataCreator](../Test-Utilities/LoggerMockDataCreator) + +[LoggerTestConfigurator](../Test-Utilities/LoggerTestConfigurator) + +--- + +### Methods + +#### `getDatabase()` → `MockDatabase` + +#### `getEventBus()` → `MockEventBus` + +#### `getJobQueue()` → `MockJobQueue` + +--- + +### Inner Classes + +#### LoggerMockDataStore.MockDatabase class + +--- + +##### Methods + +###### `insertRecords(List records)` → `List` + +--- + +#### LoggerMockDataStore.MockEventBus class + +--- + +##### Methods + +###### `deliver()` → `void` + +###### `deliver(Schema.SObjectType sobjectType)` → `void` + +###### `deliver(LoggerSObjectHandler sobjectHandlerInstance)` → `void` + +###### `getPublishCallCount()` → `Integer` + +###### `getPublishedPlatformEvents()` → `List` + +###### `publishRecord(SObject platformEvent)` → `Database.SaveResult` + +###### `publishRecords(List platformEvents)` → `List` + +--- + +#### LoggerMockDataStore.MockJobQueue class + +--- + +##### Methods + +###### `enqueueJob(Queueable queueableJob)` → `Id` + +###### `executeJobs()` → `void` + +###### `getEnqueuedJobs()` → `List` + +--- diff --git a/docs/apex/Test-Utilities/LoggerTestConfigurator.md b/docs/apex/Test-Utilities/LoggerTestConfigurator.md new file mode 100644 index 000000000..022dc8d01 --- /dev/null +++ b/docs/apex/Test-Utilities/LoggerTestConfigurator.md @@ -0,0 +1,163 @@ +--- +layout: default +--- + +## LoggerTestConfigurator class + +Utility class used to help with setting up Nebula Logger's configurations within a test context. These methods are specific to metadata implemented within Nebula Logger. These methods can be used when writing Apex tests for plugins. + +### Related + +[LoggerMockDataCreator](../Test-Utilities/LoggerMockDataCreator) + +[LoggerMockDataStore](../Test-Utilities/LoggerMockDataStore) + +--- + +### Methods + +#### `assignAdminPermissionSet(Id userId)` → `void` + +Assigns the permission set `LoggerAdmin` to the specified user ID + +##### Parameters + +| Param | Description | +| -------- | ------------------------------------------------------------- | +| `userId` | The ID of the user that should be assigned the permission set | + +#### `assignEndUserPermissionSet(Id userId)` → `void` + +Assigns the permission set `LoggerEndUser` to the specified user ID + +##### Parameters + +| Param | Description | +| -------- | ------------------------------------------------------------- | +| `userId` | The ID of the user that should be assigned the permission set | + +#### `assignLogCreatorPermissionSet(Id userId)` → `void` + +Assigns the permission set `LoggerLogCreator` to the specified user ID + +##### Parameters + +| Param | Description | +| -------- | ------------------------------------------------------------- | +| `userId` | The ID of the user that should be assigned the permission set | + +#### `assignLogViewerPermissionSet(Id userId)` → `void` + +Assigns the permission set `LoggerLogViewer` to the specified user ID + +##### Parameters + +| Param | Description | +| -------- | ------------------------------------------------------------- | +| `userId` | The ID of the user that should be assigned the permission set | + +#### `getSObjectHandlerConfiguration(Schema.SObjectType sobjectType)` → `LoggerSObjectHandler_t` + +Returns an instance of `LoggerSObjectHandler_t` that has been built & configured for the specified `SObjectType` + +##### Parameters + +| Param | Description | +| ------------- | --------------------------------------------------------------------------------------- | +| `sobjectType` | The instance `SObjectType` to check for a configured instance of `LoggerSObjectHandler` | + +##### Return + +**Type** + +LoggerSObjectHandler_t + +**Description** + +The ``LoggerSObjectHandler_t` record that has been configured for the specified `SObjectType` (if any) + +#### `setMock(LogEntryDataMaskRule_t mock)` → `void` + +Loads the mock `LogEntryDataMaskRule_t` during test execution + +##### Parameters + +| Param | Description | +| ------ | ----------------------------------------------------- | +| `mock` | The mock instance of `LogEntryDataMaskRule_t` to load | + +#### `setMock(LogEntryTagRule_t mock)` → `void` + +Loads the mock `LogEntryTagRule_t` during test execution + +##### Parameters + +| Param | Description | +| ------ | ------------------------------------------------ | +| `mock` | The mock instance of `LogEntryTagRule_t` to load | + +#### `setMock(LoggerParameter_t mock)` → `void` + +Loads the mock `LoggerParameter_t` during test execution + +##### Parameters + +| Param | Description | +| ------ | ------------------------------------------------ | +| `mock` | The mock instance of `LoggerParameter_t` to load | + +#### `setMock(LoggerPlugin_t mock)` → `void` + +Loads the mock `LoggerPlugin_t` during test execution + +##### Parameters + +| Param | Description | +| ------ | --------------------------------------------- | +| `mock` | The mock instance of `LoggerPlugin_t` to load | + +#### `setMock(LoggerSObjectHandler_t mock)` → `void` + +Loads the mock `LoggerSObjectHandler_t` during test execution + +##### Parameters + +| Param | Description | +| ------ | ----------------------------------------------------- | +| `mock` | The mock instance of `LoggerSObjectHandler_t` to load | + +#### `setMock(LogScenarioRule_t mock)` → `void` + +Loads the mock `LogScenarioRule_t` during test execution + +##### Parameters + +| Param | Description | +| ------ | ------------------------------------------------ | +| `mock` | The mock instance of `LogScenarioRule_t` to load | + +#### `setMock(LogStatus_t mock)` → `void` + +Loads the mock `LogStatus_t` during test execution + +##### Parameters + +| Param | Description | +| ------ | ------------------------------------------ | +| `mock` | The mock instance of `LogStatus_t` to load | + +#### `setupMockSObjectHandlerConfigurations()` → `void` + +Creates mock instances of `LoggerSObjectHandler_t` for each `SObjectType` used by Nebula Logger, with each `LoggerSObjectHandler_t` automatically set to `IsEnabled__c == true` + +#### `setupMockSObjectHandlerConfigurations(Boolean isEnabled)` → `void` + +Creates mock instances of `LoggerSObjectHandler_t` for each `SObjectType` used by Nebula Logger, with each `LoggerSObjectHandler_t` enabled/disabled based on the provided boolean + +##### Parameters + +| Param | Description | +| ----------- | -------------------------------------------------------------------------------------------------------------------------- | +| `isEnabled` | The Boolean value to control if all mock `LoggerSObjectHandler_t` records should be enabled (`true`) or disabled (`false`) | + +--- diff --git a/docs/apex/index.md b/docs/apex/index.md index 702b14b3f..79d03b163 100644 --- a/docs/apex/index.md +++ b/docs/apex/index.md @@ -6,104 +6,128 @@ layout: default ## Logger Engine -### [ComponentLogger](logger-engine/ComponentLogger) +### [ComponentLogger](Logger-Engine/ComponentLogger) Controller class used by the lightning web component `logger` -### [FlowCollectionLogEntry](logger-engine/FlowCollectionLogEntry) +### [FlowCollectionLogEntry](Logger-Engine/FlowCollectionLogEntry) Handles adding new log entries in Flow for a particular `SObject` record collection -### [FlowLogEntry](logger-engine/FlowLogEntry) +### [FlowLogEntry](Logger-Engine/FlowLogEntry) Handles adding new log entries in Flow -### [FlowLogger](logger-engine/FlowLogger) +### [FlowLogger](Logger-Engine/FlowLogger) Handles some common logic used by `FlowLogEntry`, `FlowRecordLogEntry` and `FlowCollectionLogEntry` -### [FlowRecordLogEntry](logger-engine/FlowRecordLogEntry) +### [FlowRecordLogEntry](Logger-Engine/FlowRecordLogEntry) Handles adding new log entries in Flow for a particular `SObject` record -### [LogEntryEventBuilder](logger-engine/LogEntryEventBuilder) +### [LogEntryEventBuilder](Logger-Engine/LogEntryEventBuilder) Builder class that generates each `LogEntryEvent__e` record -### [LogMessage](logger-engine/LogMessage) +### [LogMessage](Logger-Engine/LogMessage) Provides the ability to generate string messages on demand, using String.format() -### [Logger](logger-engine/Logger) +### [Logger](Logger-Engine/Logger) The core class for logging +### [LoggerDataStore](Logger-Engine/LoggerDataStore) + +Class used to manage any data-related operations, including database DML statements, publishing platform events via the event bus, and enqueueing queueable jobs + +### [LoggerSObjectHandler](Logger-Engine/LoggerSObjectHandler) + +Abstract class used by trigger handlers for shared logic + +### [LoggerTriggerableContext](Logger-Engine/LoggerTriggerableContext) + +Class used by the logging system for trigger contextual details + ## Log Management -### [LogBatchPurgeScheduler](log-management/LogBatchPurgeScheduler) +### [LogBatchPurgeScheduler](Log-Management/LogBatchPurgeScheduler) Schedulable class used to schedule the batch job `LogBatchPurger` -### [LogBatchPurger](log-management/LogBatchPurger) +### [LogBatchPurger](Log-Management/LogBatchPurger) Batch class used to delete old logs, based on `Log__c.LogRetentionDate__c <= :System.today()` -### [LogEntryEventHandler](log-management/LogEntryEventHandler) +### [LogEntryEventHandler](Log-Management/LogEntryEventHandler) Processes `LogEntryEvent__e` platform events and normalizes the data into `Log__c` and `LogEntry__c` records -### [LogEntryFieldSetPicklist](log-management/LogEntryFieldSetPicklist) +### [LogEntryFieldSetPicklist](Log-Management/LogEntryFieldSetPicklist) Dynamically returns `LogEntry__c` field sets in App Builder when configuring the component RelatedLogEntries -### [LogEntryHandler](log-management/LogEntryHandler) +### [LogEntryHandler](Log-Management/LogEntryHandler) Manages setting fields on `LogEntry__c` before insert & before update -### [LogEntryTagHandler](log-management/LogEntryTagHandler) +### [LogEntryTagHandler](Log-Management/LogEntryTagHandler) Handles trigger events for the `LogEntryTag__c` object -### [LogHandler](log-management/LogHandler) +### [LogHandler](Log-Management/LogHandler) Manages setting fields on `Log__c` before insert & before update -### [LogMassDeleteExtension](log-management/LogMassDeleteExtension) +### [LogMassDeleteExtension](Log-Management/LogMassDeleteExtension) Manages mass deleting `Log__c` records that have been selected by a user on a `Log__c` list view -### [LoggerSObjectHandler](log-management/LoggerSObjectHandler) +### [LoggerBatchableContext](Log-Management/LoggerBatchableContext) -Abstract class used by trigger handlers for shared logic +Class used by the logging system for batch contextual details + +### [LoggerEmailSender](Log-Management/LoggerEmailSender) + +Builds and sends email notifications when internal exceptions occur within the logging system -### [LoggerSObjectMetadata](log-management/LoggerSObjectMetadata) +### [LoggerSObjectMetadata](Log-Management/LoggerSObjectMetadata) Provides details to LWCs about Logger's `SObjects`, using `@AuraEnabled` properties -### [LoggerSettingsController](log-management/LoggerSettingsController) +### [LoggerSettingsController](Log-Management/LoggerSettingsController) Controller class for lwc `loggerSettings`, used to manage records in `LoggerSettings__c` -### [LoggerTagHandler](log-management/LoggerTagHandler) +### [LoggerTagHandler](Log-Management/LoggerTagHandler) Handles trigger events for the `LoggerTag__c` object -### [RelatedLogEntriesController](log-management/RelatedLogEntriesController) +### [RelatedLogEntriesController](Log-Management/RelatedLogEntriesController) Controller class for the lightning web component `related-log-entries` -## Configuration +## Test Utilities -### [LoggerEmailUtils](configuration/LoggerEmailUtils) +### [LoggerMockDataCreator](/Test-Utilities/LoggerMockDataCreator) -Builds and sends email notifications when internal exceptions occur within the logging system +Utility class used to help with generating mock data when writing Apex tests for Nebula Logger. These methods are generic, and should work in any Salesforce org. These methods can be used when writing Apex tests for plugins. -### [LoggerParameter](configuration/LoggerParameter) +### [LoggerMockDataStore](/Test-Utilities/LoggerMockDataStore) -Provides a centralized way to load parameters for SObject handlers & plugins, and casts the parameters to common data types +Utility class used to mock any data-related operations for the database, event bus, and queueable jobs. These methods are generic, and should work in any Salesforce org. These methods can be used when writing Apex tests for plugins. + +### [LoggerTestConfigurator](/Test-Utilities/LoggerTestConfigurator) -## Plugin Framework +Utility class used to help with setting up Nebula Logger's configurations within a test context. These methods are specific to metadata implemented within Nebula Logger. These methods can be used when writing Apex tests for plugins. + +## Configuration + +### [LoggerParameter](Configuration/LoggerParameter) + +Provides a centralized way to load parameters for SObject handlers & plugins, and casts the parameters to common data types -### [LoggerSObjectHandlerPlugin](plugin-framework/LoggerSObjectHandlerPlugin) +### [LoggerPlugin](Configuration/LoggerPlugin) -Abstract class used to create custom Apex plugins to execute for all trigger operations on `Log__c` or `LogEntry__c` +The core of the plugin framework, used to create custom Apex & Flow plugins for `LoggerSObjectHandler` and `LogBatchPurger` based on configurations stored in the custom metadata type `LoggerPlugin_t` diff --git a/docs/apex/log-management/LogEntryEventHandler.md b/docs/apex/log-management/LogEntryEventHandler.md deleted file mode 100644 index c7bddfc77..000000000 --- a/docs/apex/log-management/LogEntryEventHandler.md +++ /dev/null @@ -1,59 +0,0 @@ ---- -layout: default ---- - -## LogEntryEventHandler class - -Processes `LogEntryEvent__e` platform events and normalizes the data into `Log__c` and `LogEntry__c` records - ---- - -### Properties - -#### `releaseNumber` → `String` - -String containing the release number. - -#### `releaseVersion` → `String` - -String containing the release version. - ---- - -### Methods - -#### `executeAfterInsert(List triggerNew)` → `void` - -Public method so it can be called from Logger - this is unique to LogEntryEventHandler - -##### Parameters - -| Param | Description | -| ------------ | -------------------------------- | -| `triggerNew` | list of new LogEntryEvent events | - -#### `executeBeforeInsert(List triggerNew)` → `void` - -method so it can be called from Logger - this is unique to LogEntryEventHandler - -##### Parameters - -| Param | Description | -| ------------ | -------------------------------- | -| `triggerNew` | list of new LogEntryEvent events | - -#### `getSObjectType()` → `SObjectType` - -Returns SObject Type that the handler is responsible for processing - -##### Return - -**Type** - -SObjectType - -**Description** - -The instance of `SObjectType` - ---- diff --git a/docs/apex/log-management/LoggerSObjectHandler.md b/docs/apex/log-management/LoggerSObjectHandler.md deleted file mode 100644 index 3e0d0598a..000000000 --- a/docs/apex/log-management/LoggerSObjectHandler.md +++ /dev/null @@ -1,37 +0,0 @@ ---- -layout: default ---- - -## LoggerSObjectHandler class - -Abstract class used by trigger handlers for shared logic - ---- - -### Constructors - -#### `LoggerSObjectHandler()` - -## Default constructor - -### Methods - -#### `execute()` → `void` - -Runs the handler class's logic - -#### `getSObjectType()` → `SObjectType` - -Returns the SObject Type that the handler is responsible for processing - -##### Return - -**Type** - -SObjectType - -**Description** - -The instance of `SObjectType` - ---- diff --git a/docs/apex/log-management/LoggerSObjectMetadata.md b/docs/apex/log-management/LoggerSObjectMetadata.md deleted file mode 100644 index 6fe5469e7..000000000 --- a/docs/apex/log-management/LoggerSObjectMetadata.md +++ /dev/null @@ -1,85 +0,0 @@ ---- -layout: default ---- - -## LoggerSObjectMetadata class - -Provides details to LWCs about Logger's `SObjects`, using `@AuraEnabled` properties - ---- - -### Methods - -#### `getLogEntryEventSchema()` → `SObjectSchema` - -Provides schema details about the the platform event object `LogEntryEvent__e` - -##### Return - -**Type** - -SObjectSchema - -**Description** - -An instance of `LoggerSObjectMetadata.SObjectSchema` for the platform event `LogEntryEvent__e` - -#### `getLoggerSettingsSchema()` → `SObjectSchema` - -Provides schema details about the the custom settings object `LoggerSettings__c` - -##### Return - -**Type** - -SObjectSchema - -**Description** - -An instance of `LoggerSObjectMetadata.SObjectSchema` for the platform event `LoggerSettings__c` - ---- - -### Inner Classes - -#### LoggerSObjectMetadata.FieldSchema class - -Inner class for `SObjectField` details to LWCs, using `@AuraEnabled` properties - ---- - -##### Properties - -###### `apiName` → `String` - -###### `inlineHelpText` → `String` - -###### `label` → `String` - -###### `localApiName` → `String` - -###### `type` → `String` - ---- - -#### LoggerSObjectMetadata.SObjectSchema class - -Inner class for `SObject` details to LWCs, using `@AuraEnabled` properties - ---- - -##### Properties - -###### `apiName` → `String` - -###### `fields` → `Map` - -###### `label` → `String` - -###### `labelPlural` → `String` - -###### `localApiName` → `String` - -###### `namespacePrefix` → `String` - ---- diff --git a/docs/apex/plugin-framework/LoggerSObjectHandlerPlugin.md b/docs/apex/plugin-framework/LoggerSObjectHandlerPlugin.md deleted file mode 100644 index dbe008d41..000000000 --- a/docs/apex/plugin-framework/LoggerSObjectHandlerPlugin.md +++ /dev/null @@ -1,33 +0,0 @@ ---- -layout: default ---- - -## LoggerSObjectHandlerPlugin class - -Abstract class used to create custom Apex plugins to execute for all trigger operations on `Log__c` or `LogEntry__c` - ---- - -### Constructors - -#### `LoggerSObjectHandlerPlugin()` - -## All instances of `LoggerSObjectHandlerPlugin` are dynamically created, which requires a parameterless constructor - -### Methods - -#### `execute(TriggerOperation triggerOperationType,List triggerNew,Map triggerNewMap,List triggerOld,Map triggerOldMap)` → `void` - -This method is the entry point for plugins to execute any custom logic. It is automatically called by the logging system for any enabled plugins. Several trigger-based parameters are provided - these parameters should be used by plugins, instead of calling the platform's static variables directly (e.g., use the provided `triggerNew` variable instead of using `Trigger.new` directly, and so on). - -##### Parameters - -| Param | Description | -| ---------------------- | ------------------------------------------------------------------------------------------ | -| `triggerOperationType` | The enum instance of `Trigger.operationType` at the time that the handler class is created | -| `triggerNew` | The value `Trigger.new` at the time that the handler class is created | -| `triggerNewMap` | The value `Trigger.newMap` at the time that the handler class is created | -| `triggerOld` | The value `Trigger.old` at the time that the handler class is created | -| `triggerOldMap` | The value `Trigger.oldMap` at the time that the handler class is created | - ---- diff --git a/docs/lightning-components/LogEntryBuilder.md b/docs/lightning-components/LogEntryBuilder.md index 206eaef0d..7871c275e 100644 --- a/docs/lightning-components/LogEntryBuilder.md +++ b/docs/lightning-components/LogEntryBuilder.md @@ -32,7 +32,7 @@ This class is the JavaScript-equivalent of the Apex class `LogEntryBuilder` Sets the log entry event's message field -**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) +**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) **Returns**: [LogEntryBuilder](#LogEntryBuilder) - The same instance of `LogEntryBuilder`, useful for chaining methods | Param | Type | Description | @@ -45,7 +45,7 @@ Sets the log entry event's message field Sets the log entry event's record fields -**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) +**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) **Returns**: [LogEntryBuilder](#LogEntryBuilder) - The same instance of `LogEntryBuilder`, useful for chaining methods | Param | Type | Description | @@ -58,7 +58,7 @@ Sets the log entry event's record fields Sets the log entry event's record fields -**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) +**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) **Returns**: [LogEntryBuilder](#LogEntryBuilder) - The same instance of `LogEntryBuilder`, useful for chaining methods | Param | Type | Description | @@ -71,7 +71,7 @@ Sets the log entry event's record fields Sets the log entry event's exception fields -**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) +**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) **Returns**: [LogEntryBuilder](#LogEntryBuilder) - The same instance of `LogEntryBuilder`, useful for chaining methods | Param | Type | Description | @@ -84,7 +84,7 @@ Sets the log entry event's exception fields Appends the tag to the existing list of tags -**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) +**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) **Returns**: [LogEntryBuilder](#LogEntryBuilder) - The same instance of `LogEntryBuilder`, useful for chaining methods | Param | Type | Description | @@ -97,7 +97,7 @@ Appends the tag to the existing list of tags Appends the tag to the existing list of tags -**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) +**Kind**: instance method of [LogEntryBuilder](#LogEntryBuilder) **Returns**: [LogEntryBuilder](#LogEntryBuilder) - The same instance of `LogEntryBuilder`, useful for chaining methods | Param | Type | Description | diff --git a/docs/lightning-components/Logger.md b/docs/lightning-components/Logger.md index 6817a0baf..3c5427ea2 100644 --- a/docs/lightning-components/Logger.md +++ b/docs/lightning-components/Logger.md @@ -47,8 +47,8 @@ Returns information about the current user's settings, stored in `LoggerSettings__c` -**Kind**: global function -**Returns**: ComponentLogger.ComponentLoggerSettings - The current user's instance of the Apex class `ComponentLogger.ComponentLoggerSettings` +**Kind**: global function +**Returns**: ComponentLogger.ComponentLoggerSettings - The current user's instance of the Apex class `ComponentLogger.ComponentLoggerSettings` ## setScenario(scenario) @@ -68,7 +68,7 @@ and `Log__c.Scenario__c`, and can be used to filter & group logs Creates a new log entry with logging level == `LoggingLevel.ERROR` -**Kind**: global function +**Kind**: global function **Returns**: LogEntryBuilder - The new entry's instance of `LogEntryEventBuilder`, useful for chaining methods | Param | Type | Description | @@ -81,7 +81,7 @@ Creates a new log entry with logging level == `LoggingLevel.ERROR` Creates a new log entry with logging level == `LoggingLevel.WARN` -**Kind**: global function +**Kind**: global function **Returns**: LogEntryBuilder - The new entry's instance of `LogEntryEventBuilder`, useful for chaining methods | Param | Type | Description | @@ -94,7 +94,7 @@ Creates a new log entry with logging level == `LoggingLevel.WARN` Creates a new log entry with logging level == `LoggingLevel.INFO` -**Kind**: global function +**Kind**: global function **Returns**: LogEntryBuilder - The new entry's instance of `LogEntryEventBuilder`, useful for chaining methods | Param | Type | Description | @@ -107,7 +107,7 @@ Creates a new log entry with logging level == `LoggingLevel.INFO` Creates a new log entry with logging level == `LoggingLevel.DEBUG` -**Kind**: global function +**Kind**: global function **Returns**: LogEntryBuilder - The new entry's instance of `LogEntryEventBuilder`, useful for chaining methods | Param | Type | Description | @@ -120,7 +120,7 @@ Creates a new log entry with logging level == `LoggingLevel.DEBUG` Creates a new log entry with logging level == `LoggingLevel.FINE` -**Kind**: global function +**Kind**: global function **Returns**: LogEntryBuilder - The new entry's instance of `LogEntryEventBuilder`, useful for chaining methods | Param | Type | Description | @@ -133,7 +133,7 @@ Creates a new log entry with logging level == `LoggingLevel.FINE` Creates a new log entry with logging level == `LoggingLevel.FINER` -**Kind**: global function +**Kind**: global function **Returns**: LogEntryBuilder - The new entry's instance of `LogEntryEventBuilder`, useful for chaining methods | Param | Type | Description | @@ -146,7 +146,7 @@ Creates a new log entry with logging level == `LoggingLevel.FINER` Creates a new log entry with logging level == `LoggingLevel.FINEST` -**Kind**: global function +**Kind**: global function **Returns**: LogEntryBuilder - The new entry's instance of `LogEntryEventBuilder`, useful for chaining methods | Param | Type | Description | @@ -159,15 +159,15 @@ Creates a new log entry with logging level == `LoggingLevel.FINEST` Returns the number of entries that have been generated but not yet saved -**Kind**: global function -**Returns**: Integer - The buffer's current size +**Kind**: global function +**Returns**: Integer - The buffer's current size ## flushBuffer() Discards any entries that have been generated but not yet saved -**Kind**: global function +**Kind**: global function ## saveLog(saveMethod) diff --git a/nebula-logger/core/main/configuration/classes/LoggerParameter.cls b/nebula-logger/core/main/configuration/classes/LoggerParameter.cls index 70d5f6d30..d6c88f932 100644 --- a/nebula-logger/core/main/configuration/classes/LoggerParameter.cls +++ b/nebula-logger/core/main/configuration/classes/LoggerParameter.cls @@ -10,8 +10,10 @@ */ @SuppressWarnings('PMD.CyclomaticComplexity, PMD.ExcessivePublicCount, PMD.PropertyNamingConventions') public class LoggerParameter { - private static Set parametersToLoadDuringTests = new Set(); - private static Map mockParameterByDeveloperName = new Map(); + // During tests, always load this CMDT record + // so that tests use the same format when calling System.debug() + private static final String SYSTEM_DEBUG_MESSAGE_FORMAT_PARAMETER = 'SystemDebugMessageFormat'; + private static final Set PARAMETERS_TO_LOAD_DURING_TESTS = new Set{ SYSTEM_DEBUG_MESSAGE_FORMAT_PARAMETER }; /** * @description Indicates if Logger will make an async callout to https://api.status.salesforce.com @@ -63,11 +65,7 @@ public class LoggerParameter { public static final String SYSTEM_DEBUG_MESSAGE_FORMAT { get { if (SYSTEM_DEBUG_MESSAGE_FORMAT == null) { - // During tests, always load this CMDT record - // so that tests use the same format when calling System.debug() - String systemDebugMessageFormatParameter = 'SystemDebugMessageFormat'; - parametersToLoadDuringTests.add(systemDebugMessageFormatParameter); - SYSTEM_DEBUG_MESSAGE_FORMAT = getString(systemDebugMessageFormatParameter, '{OriginLocation__c}\n{Message__c}'); + SYSTEM_DEBUG_MESSAGE_FORMAT = getString(SYSTEM_DEBUG_MESSAGE_FORMAT_PARAMETER, '{OriginLocation__c}\n{Message__c}'); } return SYSTEM_DEBUG_MESSAGE_FORMAT; } @@ -101,6 +99,16 @@ public class LoggerParameter { private set; } + private static final Map DEVELOPER_NAME_TO_RECORD { + get { + if (DEVELOPER_NAME_TO_RECORD == null) { + DEVELOPER_NAME_TO_RECORD = loadRecords(); + } + return DEVELOPER_NAME_TO_RECORD; + } + private set; + } + /** * @description Returns the configured value of the field `Value__c` as a `Boolean` * @param parameterDeveloperName The developer name of the instance of `LoggerParameter__mdt` @@ -340,10 +348,45 @@ public class LoggerParameter { return parameterValue != null ? parameterValue : defaultValue; } - // Private methods + /** + * @description matchOnPrefix description + * @param developerNamePrefix A prefix that has been used in the `DeveloperName` for multiple `LoggerParameter__mdt` records + * @return The list of matching `LoggerParameter__mdt` records + */ + public static List matchOnPrefix(String developerNamePrefix) { + List matchingParameters = new List(); + for (String parameterDeveloperName : DEVELOPER_NAME_TO_RECORD.keySet()) { + if (parameterDeveloperName.startsWith(developerNamePrefix) == true) { + matchingParameters.add(DEVELOPER_NAME_TO_RECORD.get(parameterDeveloperName)); + } + } + return matchingParameters; + } + + private static Map loadRecords() { + Map parameters = LoggerParameter__mdt.getAll().clone(); + if (System.Test.isRunningTest() == true) { + // Keep a copy of any records that *should* be loaded during tests + // Currently, only the record `SystemDebugMessageFormat` has a use case for this functionality, + // but others can be easily added if other use cases are found + Map parametersToLoadDuringTests = new Map(); + for (String testContextParameterName : PARAMETERS_TO_LOAD_DURING_TESTS) { + if (parameters.containsKey(testContextParameterName) == true) { + parametersToLoadDuringTests.put(testContextParameterName, parameters.get(testContextParameterName)); + } + } + parameters.clear(); + parameters.putAll(parametersToLoadDuringTests); + } + return parameters; + } + @TestVisible - private static void setMockParameter(LoggerParameter__mdt parameter) { - mockParameterByDeveloperName.put(parameter.DeveloperName, parameter); + private static void setMock(LoggerParameter__mdt parameter) { + if (String.isBlank(parameter.DeveloperName) == true) { + throw new IllegalArgumentException('DeveloperName is required on `LoggerParameter__mdt: \n' + JSON.serializePretty(parameter)); + } + DEVELOPER_NAME_TO_RECORD.put(parameter.DeveloperName, parameter); } private static Object castParameterValue(String parameterDeveloperName, Type dataType) { @@ -356,16 +399,6 @@ public class LoggerParameter { } private static String loadParameterValue(String parameterDeveloperName) { - String parameterValue = LoggerParameter__mdt.getInstance(parameterDeveloperName)?.Value__c; - - Boolean useMockParameter = - parametersToLoadDuringTests.contains(parameterDeveloperName) == false || - mockParameterByDeveloperName.containsKey(parameterDeveloperName) == true; - if (System.Test.isRunningTest() == true && useMockParameter == true) { - // During tests, don't actually use the org's CMDT records - only use mock records - parameterValue = (String) mockParameterByDeveloperName.get(parameterDeveloperName)?.get('Value__c'); - } - - return parameterValue; + return DEVELOPER_NAME_TO_RECORD.get(parameterDeveloperName)?.Value__c; } } diff --git a/nebula-logger/core/main/configuration/classes/LoggerPlugin.cls b/nebula-logger/core/main/configuration/classes/LoggerPlugin.cls new file mode 100644 index 000000000..3bdbd1fdb --- /dev/null +++ b/nebula-logger/core/main/configuration/classes/LoggerPlugin.cls @@ -0,0 +1,139 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Configuration + * @description The core of the plugin framework, used to create custom Apex & Flow plugins for `LoggerSObjectHandler` and `LogBatchPurger` + * based on configurations stored in the custom metadata type `LoggerPlugin__mdt` + */ +public without sharing class LoggerPlugin { + private static final Map DEVELOPER_NAME_TO_RECORD = loadRecords(); + + /** + * @description Interface used to create plugins that can be used within Logger's batch job `LogBatchPurger` + */ + @SuppressWarnings('PMD.ApexDoc') + public interface Batchable { + void start(LoggerPlugin__mdt configuration, LoggerBatchableContext input); + void execute(LoggerPlugin__mdt configuration, LoggerBatchableContext input, List scopeRecords); + void finish(LoggerPlugin__mdt configuration, LoggerBatchableContext input); + } + + /** + * @description Interface used to create plugins that can be used within Logger's trigger handler framework `LoggerSObjectHandler` + */ + @SuppressWarnings('PMD.ApexDoc') + public interface Triggerable { + void execute(LoggerPlugin__mdt configuration, LoggerTriggerableContext input); + } + + /** + * @description Filters the configured `LoggerPlugin__mdt` records based on a list of `SObjectField` - only records that have a value for 1 or more + * of the specified `populatedFilterFields` will be returned, sorted by the specified `SObjectField` parameter `sortByField` + * @param populatedFilterFields The list of `SObjectField` to check on each `LoggerPlugin__mdt` record - filtering logic checks for a non-null value + * @param sortByField The `SObjectField` to use to sort the list of matches. The method also uses `DeveloperName` as a secondary field for sorting. + * @return The list of matching `LoggerPlugin__mdt` records + */ + public static List getFilteredPluginConfigurations(List populatedFilterFields, Schema.SObjectField sortByField) { + List matchingPluginConfigurationSorters = new List(); + for (LoggerPlugin__mdt pluginConfiguration : DEVELOPER_NAME_TO_RECORD.values()) { + Boolean matchesFilterFields = false; + for (Schema.SObjectField filterField : populatedFilterFields) { + if (pluginConfiguration.get(filterField) != null) { + matchesFilterFields = true; + break; + } + } + if (matchesFilterFields == true) { + matchingPluginConfigurationSorters.add( + new PluginConfigurationSorter(pluginConfiguration).sortBy(sortByField).sortBy(Schema.LoggerPlugin__mdt.DeveloperName) + ); + } + } + matchingPluginConfigurationSorters.sort(); + List matchingPluginConfigurations = new List(); + for (PluginConfigurationSorter sorter : matchingPluginConfigurationSorters) { + matchingPluginConfigurations.add(sorter.pluginConfiguration); + } + return matchingPluginConfigurations; + } + + /** + * @description Creates an instance of the class `LoggerPlugin.Batchable` based on the provided `LoggerPlugin__mdt` configuration + * @param apexClassTypeName The name of the Apex class that implements `LoggerPlugin.Batchable` + * @return The dynamically created instance of `LoggerPlugin.Batchable`, + * or null if an instance could not be created based on the provided configuration + */ + public static Batchable newBatchableInstance(String apexClassTypeName) { + return (Batchable) Type.forName(apexClassTypeName)?.newInstance(); + } + + /** + * @description Creates an instance of the class `LoggerPlugin.Triggerable` based on the provided `LoggerPlugin__mdt` configuration + * @param apexClassTypeName The name of the Apex class that implements `LoggerPlugin.Triggerable` + * @return The dynamically created instance of `LoggerPlugin.Triggerable`, + * or null if an instance could not be created based on the provided configuration + */ + public static Triggerable newTriggerableInstance(String apexClassTypeName) { + return (Triggerable) Type.forName(apexClassTypeName)?.newInstance(); + } + + private static Map loadRecords() { + Map pluginConfigurations = LoggerPlugin__mdt.getAll().clone(); + if (System.Test.isRunningTest() == true) { + pluginConfigurations.clear(); + } + return pluginConfigurations; + } + + @TestVisible + private static void setMock(LoggerPlugin__mdt pluginConfiguration) { + if (String.isBlank(pluginConfiguration.DeveloperName) == true) { + throw new IllegalArgumentException('DeveloperName is required on mock LoggerPlugin__mdt: \n' + JSON.serializePretty(pluginConfiguration)); + } + if (pluginConfiguration.IsEnabled__c == true) { + DEVELOPER_NAME_TO_RECORD.put(pluginConfiguration.DeveloperName, pluginConfiguration); + } + } + + @SuppressWarnings('PMD.ApexDoc') + private class PluginConfigurationSorter implements Comparable { + public LoggerPlugin__mdt pluginConfiguration; + private List sortByFields = new List(); + + public PluginConfigurationSorter(LoggerPlugin__mdt pluginConfiguration) { + this.pluginConfiguration = pluginConfiguration; + } + + public PluginConfigurationSorter sortBy(Schema.SObjectField field) { + sortByFields.add(field); + return this; + } + + public Integer compareTo(Object compareTo) { + PluginConfigurationSorter that = (PluginConfigurationSorter) compareTo; + + for (Schema.SObjectField field : this.sortByFields) { + // Ugly block to handle numeric comparisons vs string comparisons (based on the field type) + Boolean thisIsGreaterThanThat = false; + if (field.getDescribe().getSoapType() == Schema.SoapType.DOUBLE) { + thisIsGreaterThanThat = (Decimal) this.pluginConfiguration.get(field) > (Decimal) that.pluginConfiguration.get(field); + } else { + thisIsGreaterThanThat = (String) this.pluginConfiguration.get(field) > (String) that.pluginConfiguration.get(field); + } + + // Now, the actual comparisons + if (this.pluginConfiguration.get(field) == that.pluginConfiguration.get(field)) { + continue; + } else if (this.pluginConfiguration.get(field) == null && that.pluginConfiguration.get(field) != null || thisIsGreaterThanThat == true) { + return 1; + } else { + return -1; + } + } + return 0; + } + } +} diff --git a/nebula-logger/core/main/configuration/classes/LoggerEmailUtils.cls-meta.xml b/nebula-logger/core/main/configuration/classes/LoggerPlugin.cls-meta.xml similarity index 100% rename from nebula-logger/core/main/configuration/classes/LoggerEmailUtils.cls-meta.xml rename to nebula-logger/core/main/configuration/classes/LoggerPlugin.cls-meta.xml diff --git a/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryEventHandlerEnabled.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryEventHandlerEnabled.md-meta.xml new file mode 100644 index 000000000..d09af9772 --- /dev/null +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryEventHandlerEnabled.md-meta.xml @@ -0,0 +1,17 @@ + + + + false + + Description__c + Controls if the trigger handler class LogEntryEventHandler executes for LogEntryEvent__e records + + + Value__c + true + + diff --git a/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryHandlerEnabled.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryHandlerEnabled.md-meta.xml new file mode 100644 index 000000000..e7e66e4c0 --- /dev/null +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryHandlerEnabled.md-meta.xml @@ -0,0 +1,17 @@ + + + + false + + Description__c + Controls if the trigger handler class LogEntryHandler executes for LogEntry__c records + + + Value__c + true + + diff --git a/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryTagHandlerEnabled.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryTagHandlerEnabled.md-meta.xml new file mode 100644 index 000000000..ba3dff458 --- /dev/null +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogEntryTagHandlerEnabled.md-meta.xml @@ -0,0 +1,17 @@ + + + + false + + Description__c + Controls if the trigger handler class LogEntryTagHandler executes for LogEntryTag__c records + + + Value__c + true + + diff --git a/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogHandlerEnabled.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogHandlerEnabled.md-meta.xml new file mode 100644 index 000000000..86069e05b --- /dev/null +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLogHandlerEnabled.md-meta.xml @@ -0,0 +1,17 @@ + + + + false + + Description__c + Controls if the trigger handler class LogEntryHandler executes for LogEntry__c records + + + Value__c + true + + diff --git a/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLoggerTagHandlerEnabled.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLoggerTagHandlerEnabled.md-meta.xml new file mode 100644 index 000000000..77d91950b --- /dev/null +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.IsLoggerTagHandlerEnabled.md-meta.xml @@ -0,0 +1,17 @@ + + + + false + + Description__c + Controls if the trigger handler class LoggerTagHandler executes for LoggerTag__c records + + + Value__c + true + + diff --git a/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.SendErrorEmailNotifications.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.SendErrorEmailNotifications.md-meta.xml index 2261d1690..271e59d0b 100644 --- a/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.SendErrorEmailNotifications.md-meta.xml +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerParameter.SendErrorEmailNotifications.md-meta.xml @@ -10,7 +10,7 @@ Description__c When set to 'true' (default), if an internal error occurs within Logger, an email notification will be sent to the org's list of Apex Exception Email recipients, configured under Setup --> Email --> Apex Exception Email. + >When set to 'true' (default), if an internal error occurs within Logger, an email notification will be sent to the org's list of Apex Exception Email recipients, configured under Setup --> Email --> Apex Exception Email. When set to 'false', no emails will be sent by Logger. diff --git a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryEventHandler.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryEventHandler.md-meta.xml similarity index 63% rename from nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryEventHandler.md-meta.xml rename to nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryEventHandler.md-meta.xml index 1bc52dc7f..f6c7d3e37 100644 --- a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryEventHandler.md-meta.xml +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryEventHandler.md-meta.xml @@ -6,6 +6,14 @@ > false + + IsEnabled__c + true + + + SObjectHandlerApexClass__c + LogEntryEventHandler + SObjectType__c LogEntryEvent__e diff --git a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryHandler.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryHandler.md-meta.xml similarity index 63% rename from nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryHandler.md-meta.xml rename to nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryHandler.md-meta.xml index 1fa2d72d0..5cb69a7c2 100644 --- a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryHandler.md-meta.xml +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryHandler.md-meta.xml @@ -6,6 +6,14 @@ > false + + IsEnabled__c + true + + + SObjectHandlerApexClass__c + LogEntryHandler + SObjectType__c LogEntry__c diff --git a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryTagHandler.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryTagHandler.md-meta.xml similarity index 63% rename from nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryTagHandler.md-meta.xml rename to nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryTagHandler.md-meta.xml index ca5175002..749122218 100644 --- a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogEntryTagHandler.md-meta.xml +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogEntryTagHandler.md-meta.xml @@ -6,6 +6,14 @@ > false + + IsEnabled__c + true + + + SObjectHandlerApexClass__c + LogEntryTagHandler + SObjectType__c LogEntryTag__c diff --git a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogHandler.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogHandler.md-meta.xml similarity index 63% rename from nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogHandler.md-meta.xml rename to nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogHandler.md-meta.xml index 77c2ad016..93ed4638a 100644 --- a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LogHandler.md-meta.xml +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LogHandler.md-meta.xml @@ -6,6 +6,14 @@ > false + + IsEnabled__c + true + + + SObjectHandlerApexClass__c + LogHandler + SObjectType__c Log__c diff --git a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LoggerTagHandler.md-meta.xml b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LoggerTagHandler.md-meta.xml similarity index 79% rename from nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LoggerTagHandler.md-meta.xml rename to nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LoggerTagHandler.md-meta.xml index 5a1fdd053..6ee82a774 100644 --- a/nebula-logger/managed-package/core/main/deprecated/customMetadata/LoggerSObjectHandler.LoggerTagHandler.md-meta.xml +++ b/nebula-logger/core/main/configuration/customMetadata/LoggerSObjectHandler.LoggerTagHandler.md-meta.xml @@ -10,6 +10,10 @@ IsEnabled__c true + + SObjectHandlerApexClass__c + LoggerTagHandler + SObjectType__c LoggerTag__c diff --git a/nebula-logger/core/main/configuration/layouts/LogEntryDataMaskRule__mdt-Log Entry Data Mask Rule Layout.layout-meta.xml b/nebula-logger/core/main/configuration/layouts/LogEntryDataMaskRule__mdt-Log Entry Data Mask Rule Layout.layout-meta.xml index 73e37b0ba..dfc86a88f 100644 --- a/nebula-logger/core/main/configuration/layouts/LogEntryDataMaskRule__mdt-Log Entry Data Mask Rule Layout.layout-meta.xml +++ b/nebula-logger/core/main/configuration/layouts/LogEntryDataMaskRule__mdt-Log Entry Data Mask Rule Layout.layout-meta.xml @@ -20,14 +20,6 @@ Edit IsEnabled__c - - Edit - ApplyToMessage__c - - - Edit - ApplyToRecordJson__c - diff --git a/nebula-logger/core/main/configuration/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml b/nebula-logger/core/main/configuration/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml new file mode 100644 index 000000000..2e002a970 --- /dev/null +++ b/nebula-logger/core/main/configuration/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml @@ -0,0 +1,152 @@ + + + + false + true + true + + + + Required + MasterLabel + + + Required + DeveloperName + + + + + Edit + IsEnabled__c + + + + + + true + false + false + + + + Edit + Description__c + + + + + + true + true + true + + + + Edit + SObjectHandlerApexClass__c + + + Edit + SObjectHandlerFlowName__c + + + + + Edit + SObjectHandlerExecutionOrder__c + + + + + + true + true + true + + + + Edit + BatchPurgerApexClass__c + + + Edit + BatchPurgerFlowName__c + + + + + Edit + BatchPurgerExecutionOrder__c + + + + + + true + true + true + + + + Edit + VersionNumber__c + + + + + Edit + Link__c + + + + + + false + true + true + + + + Required + NamespacePrefix + + + Readonly + CreatedById + + + + + Edit + IsProtected + + + Readonly + LastModifiedById + + + + + + true + true + false + + + + + + + false + false + false + false + false + + 00h1F000006uWJh + 4 + 0 + Default + + diff --git a/nebula-logger/managed-package/core/main/deprecated/layouts/LoggerSObjectHandler__mdt-Logger SObject Handler Layout.layout-meta.xml b/nebula-logger/core/main/configuration/layouts/LoggerSObjectHandler__mdt-Logger SObject Handler Layout.layout-meta.xml similarity index 77% rename from nebula-logger/managed-package/core/main/deprecated/layouts/LoggerSObjectHandler__mdt-Logger SObject Handler Layout.layout-meta.xml rename to nebula-logger/core/main/configuration/layouts/LoggerSObjectHandler__mdt-Logger SObject Handler Layout.layout-meta.xml index 3680063db..055e5a5c6 100644 --- a/nebula-logger/managed-package/core/main/deprecated/layouts/LoggerSObjectHandler__mdt-Logger SObject Handler Layout.layout-meta.xml +++ b/nebula-logger/core/main/configuration/layouts/LoggerSObjectHandler__mdt-Logger SObject Handler Layout.layout-meta.xml @@ -14,6 +14,10 @@ Required DeveloperName + + Required + SObjectHandlerApexClass__c + @@ -64,26 +68,6 @@ - - MasterLabel - DeveloperName - IsEnabled__c - CollectionType__c - DataType__c - Value__c - LoggerSObjectHandlerParameter__mdt.SObjectHandler__c - - - MasterLabel - DeveloperName - IsEnabled__c - PluginType__c - PluginApiName__c - ExecutionOrder__c - LoggerSObjectHandlerPlugin__mdt.SObjectHandler__c - ExecutionOrder__c - Asc - false false false diff --git a/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToMessage__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToMessage__c.field-meta.xml index eac8ff40c..db3c9ad1e 100644 --- a/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToMessage__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToMessage__c.field-meta.xml @@ -1,11 +1,12 @@ ApplyToMessage__c + DeprecateCandidate true false DeveloperControlled When checked, the data mask rule is applied to log entry messages (stored in LogEntryEvent__e.Message__c and LogEntry__c.Message__c) - + Checkbox diff --git a/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToRecordJson__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToRecordJson__c.field-meta.xml index c2a7a021e..adc3eb4f7 100644 --- a/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToRecordJson__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/fields/ApplyToRecordJson__c.field-meta.xml @@ -1,6 +1,7 @@ ApplyToRecordJson__c + DeprecateCandidate true false DeveloperControlled diff --git a/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/listViews/All.listView-meta.xml b/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/listViews/All.listView-meta.xml index 857957973..4a0900227 100644 --- a/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/listViews/All.listView-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LogEntryDataMaskRule__mdt/listViews/All.listView-meta.xml @@ -4,8 +4,6 @@ MasterLabel DeveloperName IsEnabled__c - ApplyToMessage__c - ApplyToRecordJson__c SensitiveDataRegEx__c ReplacementRegEx__c Everything diff --git a/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Description__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Description__c.field-meta.xml index 12250d074..8bb81ba37 100644 --- a/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Description__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Description__c.field-meta.xml @@ -1,10 +1,12 @@ Description__c + Active false DeveloperControlled 131072 + Confidential LongTextArea 3 diff --git a/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Value__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Value__c.field-meta.xml index ef72c996a..8e9b04d62 100644 --- a/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Value__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/fields/Value__c.field-meta.xml @@ -1,10 +1,12 @@ Value__c + Active false SubscriberControlled 131072 + Confidential LongTextArea 5 diff --git a/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/listViews/All.listView-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/listViews/All.listView-meta.xml index 0a99077ac..502caccd2 100644 --- a/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/listViews/All.listView-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerParameter__mdt/listViews/All.listView-meta.xml @@ -5,6 +5,7 @@ DeveloperName Value__c Description__c + IsProtected Everything diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/LoggerPlugin__mdt.object-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/LoggerPlugin__mdt.object-meta.xml similarity index 100% rename from nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/LoggerPlugin__mdt.object-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/LoggerPlugin__mdt.object-meta.xml diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerApexClass__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerApexClass__c.field-meta.xml new file mode 100644 index 000000000..7cc5f650a --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerApexClass__c.field-meta.xml @@ -0,0 +1,11 @@ + + + BatchPurgerApexClass__c + false + DeveloperControlled + + 255 + false + Text + false + diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerExecutionOrder__c.field-meta.xml similarity index 88% rename from nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerExecutionOrder__c.field-meta.xml index 802055ce4..74960641e 100644 --- a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerExecutionOrder__c.field-meta.xml @@ -1,13 +1,13 @@ - ExecutionOrder__c + BatchPurgerExecutionOrder__c The specific order to execute plugins in the org. Plugins are executed in order by sorting by execution order, then plugin name (ORDER BY ExecutionOrder__c NULLS LAST, DeveloperName). false SubscriberControlled The specific order to execute plugins in the org. Plugins are executed in order by sorting by execution order, then plugin name (ORDER BY ExecutionOrder__c NULLS LAST, DeveloperName). - + 3 false 0 diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerFlowName__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerFlowName__c.field-meta.xml new file mode 100644 index 000000000..8bd098b7d --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/BatchPurgerFlowName__c.field-meta.xml @@ -0,0 +1,11 @@ + + + BatchPurgerFlowName__c + false + DeveloperControlled + + 255 + false + Text + false + diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/Description__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/Description__c.field-meta.xml similarity index 100% rename from nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/Description__c.field-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/Description__c.field-meta.xml diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml new file mode 100644 index 000000000..5a03d64e3 --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/ExecutionOrder__c.field-meta.xml @@ -0,0 +1,14 @@ + + + ExecutionOrder__c + Deprecated: This field is no longer used. + false + SubscriberControlled + Deprecated: This field is no longer used. + + 3 + false + 0 + Number + false + diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/IsEnabled__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/IsEnabled__c.field-meta.xml similarity index 100% rename from nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/IsEnabled__c.field-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/IsEnabled__c.field-meta.xml diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/Link__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/Link__c.field-meta.xml new file mode 100644 index 000000000..d592b2fe4 --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/Link__c.field-meta.xml @@ -0,0 +1,10 @@ + + + Link__c + false + DeveloperControlled + An optional link for documentation about this plugin - useful for plugins that are packaged or deployed to multiple orgs + + false + Url + diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml new file mode 100644 index 000000000..9a2134a40 --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml @@ -0,0 +1,13 @@ + + + PluginApiName__c + Deprecated: This field is no longer used. + false + DeveloperControlled + Deprecated: This field is no longer used. + + 255 + false + Text + false + diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/PluginType__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/PluginType__c.field-meta.xml similarity index 71% rename from nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/PluginType__c.field-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/PluginType__c.field-meta.xml index 18b1d4371..b0b966f37 100644 --- a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/PluginType__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/PluginType__c.field-meta.xml @@ -1,12 +1,13 @@ PluginType__c - The type of plugin that the logging system should dynamically execute - Apex or Flow. + DeprecateCandidate + Deprecated: This field is no longer used. false DeveloperControlled - The type of plugin that the logging system should dynamically execute - Apex or Flow. - - true + Deprecated: This field is no longer used. + + false Picklist true diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml new file mode 100644 index 000000000..9fba0a7ee --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml @@ -0,0 +1,11 @@ + + + SObjectHandlerApexClass__c + false + DeveloperControlled + + 255 + false + Text + false + diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerExecutionOrder__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerExecutionOrder__c.field-meta.xml new file mode 100644 index 000000000..8220324f8 --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerExecutionOrder__c.field-meta.xml @@ -0,0 +1,16 @@ + + + SObjectHandlerExecutionOrder__c + The specific order to execute plugins within Logger's trigger framework. Plugins are executed in order by sorting by execution order, then plugin name (ORDER BY ExecutionOrder__c NULLS LAST, DeveloperName). + false + SubscriberControlled + The specific order to execute plugins within Logger's trigger framework. Plugins are executed in order by sorting by execution order, then plugin name (ORDER BY ExecutionOrder__c NULLS LAST, DeveloperName). + + 3 + false + 0 + Number + false + diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerFlowName__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerFlowName__c.field-meta.xml new file mode 100644 index 000000000..2af5ac38d --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectHandlerFlowName__c.field-meta.xml @@ -0,0 +1,11 @@ + + + SObjectHandlerFlowName__c + false + DeveloperControlled + + 255 + false + Text + false + diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/SObjectType__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectType__c.field-meta.xml similarity index 63% rename from nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/SObjectType__c.field-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectType__c.field-meta.xml index 7e6b5482a..44b23d9c4 100644 --- a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/SObjectType__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/SObjectType__c.field-meta.xml @@ -1,14 +1,16 @@ SObjectType__c + DeprecateCandidate + Deprecated: This field is no longer used. false DeveloperControlled - The logging system's SObject that should dynamically call the plugin. This controls when the plugin is executed. - + Deprecated: This field is no longer used. + EntityDefinition Logger Plugins LoggerSObjectHandlerPlugins - true + false MetadataRelationship false diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/VersionNumber__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/VersionNumber__c.field-meta.xml new file mode 100644 index 000000000..a9d33f11d --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/fields/VersionNumber__c.field-meta.xml @@ -0,0 +1,12 @@ + + + VersionNumber__c + false + DeveloperControlled + An optional version number for the plugin - useful for plugins that are packaged or deployed to multiple orgs + + 18 + false + Text + false + diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/listViews/All.listView-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/listViews/All.listView-meta.xml new file mode 100644 index 000000000..55e23a72f --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/listViews/All.listView-meta.xml @@ -0,0 +1,13 @@ + + + All + MasterLabel + DeveloperName + IsEnabled__c + SObjectHandlerExecutionOrder__c + BatchPurgerExecutionOrder__c + VersionNumber__c + Description__c + Everything + + diff --git a/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml new file mode 100644 index 000000000..62c4d3c35 --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml @@ -0,0 +1,9 @@ + + + Plugins_are_not_supported + false + Deprecated: this validation rule is no longer used + false + IsEnabled__c + Plugins are not supported for this SObject Handler + diff --git a/nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/LoggerSObjectHandler__mdt.object-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/LoggerSObjectHandler__mdt.object-meta.xml similarity index 73% rename from nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/LoggerSObjectHandler__mdt.object-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/LoggerSObjectHandler__mdt.object-meta.xml index 138dc5541..b97113ef0 100644 --- a/nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/LoggerSObjectHandler__mdt.object-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/LoggerSObjectHandler__mdt.object-meta.xml @@ -1,7 +1,7 @@ Used to configure the Apex trigger handler classes that run on the objects LogEntryEvent__e, Log__c and LogEntry__c - - OLD: Logger SObject Handlers + + Logger SObject Handlers Public diff --git a/nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/fields/IsEnabled__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/IsEnabled__c.field-meta.xml similarity index 100% rename from nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/fields/IsEnabled__c.field-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/IsEnabled__c.field-meta.xml diff --git a/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml new file mode 100644 index 000000000..a8741b00b --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/SObjectHandlerApexClass__c.field-meta.xml @@ -0,0 +1,13 @@ + + + SObjectHandlerApexClass__c + false + false + DeveloperControlled + The Apex class that extends the abstract LoggerSObjectHandler for the specified SObject Type + + 255 + false + Text + true + diff --git a/nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/fields/SObjectType__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/SObjectType__c.field-meta.xml similarity index 100% rename from nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/fields/SObjectType__c.field-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/fields/SObjectType__c.field-meta.xml diff --git a/nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/listViews/All.listView-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/listViews/All.listView-meta.xml similarity index 87% rename from nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/listViews/All.listView-meta.xml rename to nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/listViews/All.listView-meta.xml index f5ac4c12a..ac8c47633 100644 --- a/nebula-logger/managed-package/core/main/deprecated/objects/LoggerSObjectHandler__mdt/listViews/All.listView-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerSObjectHandler__mdt/listViews/All.listView-meta.xml @@ -5,6 +5,7 @@ DeveloperName IsEnabled__c SObjectType__c + SObjectHandlerApexClass__c Everything diff --git a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogPurgeAction__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogPurgeAction__c.field-meta.xml new file mode 100644 index 000000000..a6513f00e --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogPurgeAction__c.field-meta.xml @@ -0,0 +1,14 @@ + + + DefaultLogPurgeAction__c + 'Delete' + false + Sets the value of Log__c.LogPurgeAction__c, which is used by the batch job LogBatchPurger to indicate the action that should be taken for a Log__c record when LogRetentionDate__c < TODAY. + + 255 + false + false + Text + false + diff --git a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogShareAccessLevel__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogShareAccessLevel__c.field-meta.xml index 1e0232424..faf8d70c7 100644 --- a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogShareAccessLevel__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultLogShareAccessLevel__c.field-meta.xml @@ -3,8 +3,7 @@ DefaultLogShareAccessLevel__c 'Read' Uses Apex managed sharing to grants users read or edit access to their log records (on insert only). When no access level is specified, no Apex sharing logic is executed. This only gives record-level access - users will still need to be granted access to the Log__c object using permission sets or profiles. - + >Uses Apex managed sharing to grants users read or edit access to their log records (on insert only). When no access level is specified, no Apex sharing logic is executed. This only gives record-level access - users will still need to be granted access to the Log__c object using permission sets or profiles. false Uses Apex managed sharing to grants users read or edit access to their log records (on insert only). When no access level is specified, no Apex sharing logic is executed. This only gives record-level access - users will still need to be granted access to the Log__c object using permission sets or profiles. diff --git a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultPlatformEventStorageLocation__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultPlatformEventStorageLocation__c.field-meta.xml new file mode 100644 index 000000000..329a5fa8c --- /dev/null +++ b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/DefaultPlatformEventStorageLocation__c.field-meta.xml @@ -0,0 +1,16 @@ + + + DefaultPlatformEventStorageLocation__c + 'CUSTOM_OBJECTS' + Defaults to CUSTOM_OBJECTS. This controls the default location where LogEntryEvent__e records are stored - when null, LogEntryEvent__e records will not be stored. + false + Defaults to CUSTOM_OBJECTS. This controls the default location where LogEntryEvent__e records are stored - when null, LogEntryEvent__e records will not be stored. + + 255 + false + false + Text + false + diff --git a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/IsSavingEnabled__c.field-meta.xml b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/IsSavingEnabled__c.field-meta.xml index f9be7057f..250389be8 100644 --- a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/IsSavingEnabled__c.field-meta.xml +++ b/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/IsSavingEnabled__c.field-meta.xml @@ -6,5 +6,6 @@ false Controls if saving is enabled - when disabled, any calls to saveLog() are ignored. + false Checkbox diff --git a/nebula-logger/core/main/log-management/applications/LoggerConsole.app-meta.xml b/nebula-logger/core/main/log-management/applications/LoggerConsole.app-meta.xml index c21894a87..ed1fa40be 100644 --- a/nebula-logger/core/main/log-management/applications/LoggerConsole.app-meta.xml +++ b/nebula-logger/core/main/log-management/applications/LoggerConsole.app-meta.xml @@ -9,6 +9,7 @@ Large false false + false Console Log__c diff --git a/nebula-logger/core/main/log-management/classes/LogBatchPurgeScheduler.cls b/nebula-logger/core/main/log-management/classes/LogBatchPurgeScheduler.cls index 424e55b17..9576d76c8 100644 --- a/nebula-logger/core/main/log-management/classes/LogBatchPurgeScheduler.cls +++ b/nebula-logger/core/main/log-management/classes/LogBatchPurgeScheduler.cls @@ -39,6 +39,6 @@ global with sharing class LogBatchPurgeScheduler implements System.Schedulable { Logger.saveLog(); } - Database.executebatch(new LogBatchPurger(), this.batchSize); + Database.executebatch(new LogBatchPurger().setChainedBatchSize(this.batchSize), this.batchSize); } } diff --git a/nebula-logger/core/main/log-management/classes/LogBatchPurger.cls b/nebula-logger/core/main/log-management/classes/LogBatchPurger.cls index 01121fa10..747f39549 100644 --- a/nebula-logger/core/main/log-management/classes/LogBatchPurger.cls +++ b/nebula-logger/core/main/log-management/classes/LogBatchPurger.cls @@ -8,12 +8,28 @@ * @description Batch class used to delete old logs, based on `Log__c.LogRetentionDate__c <= :System.today()` * @see LogBatchPurgeScheduler */ -@SuppressWarnings('PMD.AvoidGlobalModifier') +@SuppressWarnings('PMD.AvoidGlobalModifier, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList') global with sharing class LogBatchPurger implements Database.Batchable, Database.Stateful { + private static final Date LOG_RETENTION_END_DATE = System.today(); @TestVisible private static final String NO_DELETE_ACCESS_EXCEPTION_MESSAGE = 'User does not have access to delete logs'; + @TestVisible + private enum BatchableMethod { + START, + EXECUTE, + FINISH + } + + // Instance variables for top-level class + @TestVisible + private Integer chainedBatchSize = 200; + @TestVisible private SObjectType currentSObjectType; + private List pluginConfigurations; + private Map> methodToExecutedApexPlugins = new Map>(); + private Map> methodToExecutedFlowPlugins = new Map>(); + private String originalTransactionId; private Integer totalProcessedRecords; @@ -22,6 +38,17 @@ global with sharing class LogBatchPurger implements Database.Batchable, */ global LogBatchPurger() { this.totalProcessedRecords = 0; + this.pluginConfigurations = LoggerPlugin.getFilteredPluginConfigurations( + new List{ Schema.LoggerPlugin__mdt.BatchPurgerApexClass__c, Schema.LoggerPlugin__mdt.BatchPurgerFlowName__c }, + Schema.LoggerPlugin__mdt.BatchPurgerExecutionOrder__c + ); + + if (System.Test.isRunningTest() == true) { + for (BatchableMethod method : BatchableMethod.values()) { + this.methodToExecutedApexPlugins.put(method, new List()); + this.methodToExecutedFlowPlugins.put(method, new List()); + } + } } /** @@ -42,12 +69,16 @@ global with sharing class LogBatchPurger implements Database.Batchable, // Each batchable method runs in a separate transaction, // so store the first transaction ID to later relate the other transactions this.originalTransactionId = Logger.getTransactionId(); - this.currentSObjectType = this.getInitialSObjectType(); + + Schema.SObjectType initialSObjectType = this.getInitialSObjectType(); + LoggerBatchableContext input = new LoggerBatchableContext(batchableContext, initialSObjectType); + this.executePlugins(BatchableMethod.START, input, null); + this.currentSObjectType = input.sobjectType; if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { Logger.info('Logger - Starting LogBatchPurger job for SObject type: ' + this.currentSObjectType); - Logger.saveLog(); } + Logger.saveLog(); return this.getQueryLocator(this.currentSObjectType); } @@ -69,18 +100,23 @@ global with sharing class LogBatchPurger implements Database.Batchable, * @description Required by the Database.Batchable interface, this method executes the logic for purging * log records. * @param batchableContext - The context of the current batch job. - * @param loggerRecords - The log records to purge. + * @param scopeRecords - The log records to purge. */ - global void execute(Database.BatchableContext batchableContext, List loggerRecords) { + global void execute(Database.BatchableContext batchableContext, List scopeRecords) { + Logger.setParentLogTransactionId(this.originalTransactionId); try { - this.totalProcessedRecords += loggerRecords.size(); - this.hardDelete(loggerRecords); + this.totalProcessedRecords += scopeRecords.size(); + LoggerBatchableContext input = new LoggerBatchableContext(batchableContext, this.currentSObjectType); + this.executePlugins(BatchableMethod.EXECUTE, input, scopeRecords); + + LoggerDataStore.getDatabase().hardDeleteRecords(scopeRecords); + Logger.saveLog(); } catch (Exception apexException) { if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { Logger.error('Logger - Error deleting logs', apexException); } - } finally { Logger.saveLog(); + throw apexException; } } @@ -90,13 +126,16 @@ global with sharing class LogBatchPurger implements Database.Batchable, * @param batchableContext - The context of the batch jobs */ global void finish(Database.BatchableContext batchableContext) { + Logger.setParentLogTransactionId(this.originalTransactionId); + LoggerBatchableContext input = new LoggerBatchableContext(batchableContext, this.currentSObjectType); + this.executePlugins(BatchableMethod.FINISH, input, null); + Id nextBatchJobId; if (this.currentSObjectType != Schema.Log__c.SObjectType) { - nextBatchJobId = Database.executeBatch(new LogBatchPurger()); + nextBatchJobId = Database.executeBatch(this, this.chainedBatchSize); } if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { - Logger.setParentLogTransactionId(this.originalTransactionId); Logger.info( new LogMessage( 'Logger - Finished LogBatchPurger job for {0}, {1} total log records processed', @@ -111,14 +150,16 @@ global with sharing class LogBatchPurger implements Database.Batchable, } } - private void hardDelete(List records) { - // normally this would be an anti-pattern since most DML operations - // are a no-op with an empty list - but emptyRecycleBin throws - // for empty lists! - if (!records.isEmpty()) { - delete records; - Database.emptyRecycleBin(records); - } + /** + * @description The `LogBatchPurger` job is designed to run several instances - typically, it runs on `LogEntryTag__c`, + * then `LogEntry__c`, and finally `Log__c`. This method provides a way to control the batch size used for + * the chained instances of `LogBachPurger` + * @param chainedBatchSize The batch size to use for any subsequent chained instances of `LogBatchPurger` + * @return The same instance of `LogBatchPurger`, useful for chaining methods + */ + public LogBatchPurger setChainedBatchSize(Integer chainedBatchSize) { + this.chainedBatchSize = chainedBatchSize; + return this; } private Schema.SObjectType getInitialSObjectType() { @@ -137,12 +178,12 @@ global with sharing class LogBatchPurger implements Database.Batchable, return [ SELECT COUNT() FROM LogEntryTag__c - WHERE LogEntry__r.Log__r.LogRetentionDate__c <= :System.today() AND LogEntry__r.Log__r.LogRetentionDate__c != NULL + WHERE LogEntry__r.Log__r.LogRetentionDate__c <= :LOG_RETENTION_END_DATE AND LogEntry__r.Log__r.LogRetentionDate__c != NULL ]; } private Integer getLogEntryCount() { - return [SELECT COUNT() FROM LogEntry__c WHERE Log__r.LogRetentionDate__c <= :System.today() AND Log__r.LogRetentionDate__c != NULL]; + return [SELECT COUNT() FROM LogEntry__c WHERE Log__r.LogRetentionDate__c <= :LOG_RETENTION_END_DATE AND Log__r.LogRetentionDate__c != NULL]; } /** @@ -162,7 +203,7 @@ global with sharing class LogBatchPurger implements Database.Batchable, [ SELECT Id FROM LogEntryTag__c - WHERE LogEntry__r.Log__r.LogRetentionDate__c <= :System.today() AND LogEntry__r.Log__r.LogRetentionDate__c != NULL + WHERE LogEntry__r.Log__r.LogRetentionDate__c <= :LOG_RETENTION_END_DATE AND LogEntry__r.Log__r.LogRetentionDate__c != NULL ORDER BY LogEntry__r.Log__r.LogRetentionDate__c, LogEntry__r.Log__c ] ); @@ -172,7 +213,7 @@ global with sharing class LogBatchPurger implements Database.Batchable, [ SELECT Id FROM LogEntry__c - WHERE Log__r.LogRetentionDate__c <= :System.today() AND Log__r.LogRetentionDate__c != NULL + WHERE Log__r.LogRetentionDate__c <= :LOG_RETENTION_END_DATE AND Log__r.LogRetentionDate__c != NULL ORDER BY Log__r.LogRetentionDate__c, Log__c ] ); @@ -182,7 +223,7 @@ global with sharing class LogBatchPurger implements Database.Batchable, [ SELECT Id FROM Log__c - WHERE (LogRetentionDate__c <= :System.today() AND LogRetentionDate__c != NULL) OR TotalLogEntries__c = 0 + WHERE (LogRetentionDate__c <= :LOG_RETENTION_END_DATE AND LogRetentionDate__c != NULL) OR TotalLogEntries__c = 0 ORDER BY LogRetentionDate__c ] ); @@ -191,4 +232,84 @@ global with sharing class LogBatchPurger implements Database.Batchable, return queryLocator; } + + private void executePlugins(BatchableMethod method, LoggerBatchableContext input, List scopeRecords) { + for (LoggerPlugin__mdt pluginConfiguration : this.pluginConfigurations) { + if (String.isNotBlank(pluginConfiguration.BatchPurgerApexClass__c) == true) { + this.executeApexPlugin(method, pluginConfiguration, input, scopeRecords); + } + if (String.isNotBlank(pluginConfiguration.BatchPurgerFlowName__c) == true) { + this.executeFlowPlugin(method, pluginConfiguration, input, scopeRecords); + } + } + } + + @SuppressWarnings('PMD.AvoidDebugStatements') + private void executeApexPlugin(BatchableMethod method, LoggerPlugin__mdt pluginConfiguration, LoggerBatchableContext input, List scopeRecords) { + if (String.isBlank(pluginConfiguration.BatchPurgerApexClass__c) == true) { + return; + } + + try { + LoggerPlugin.Batchable apexPlugin = LoggerPlugin.newBatchableInstance(pluginConfiguration.BatchPurgerApexClass__c); + switch on method { + when START { + apexPlugin.start(pluginConfiguration, input); + } + when EXECUTE { + apexPlugin.execute(pluginConfiguration, input, scopeRecords); + } + when FINISH { + apexPlugin.finish(pluginConfiguration, input); + } + } + + if (System.Test.isRunningTest() == true && apexPlugin != null) { + this.methodToExecutedApexPlugins.get(method).add(apexPlugin); + } + } catch (TypeException ex) { + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.warn('Unknown Apex class ' + pluginConfiguration.BatchPurgerApexClass__c + ', skipping plugin execution', pluginConfiguration, ex); + } + } + } + + @SuppressWarnings('PMD.AvoidDebugStatements') + private void executeFlowPlugin(BatchableMethod method, LoggerPlugin__mdt pluginConfiguration, LoggerBatchableContext input, List scopeRecords) { + if (String.isBlank(pluginConfiguration.BatchPurgerFlowName__c) == true) { + return; + } + + try { + Map flowInputs = new Map(); + flowInputs.put('pluginConfiguration', pluginConfiguration); + flowInputs.put('pluginInput', input); + if (scopeRecords != null) { + flowInputs.put('scopeRecords', scopeRecords); + } + + Flow.Interview flowPlugin = Flow.Interview.createInterview(pluginConfiguration.BatchPurgerFlowName__c, flowInputs); + flowPlugin.start(); + + if (System.Test.isRunningTest() == true && flowPlugin != null) { + this.methodToExecutedFlowPlugins.get(method).add(flowPlugin); + } + } catch (TypeException ex) { + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.warn('Unknown Flow ' + pluginConfiguration.BatchPurgerFlowName__c + ', skipping plugin execution', pluginConfiguration, ex); + } + } + } + + // TODO Before v4.8.0, consider changing visibility from `@TestVisbile private` to `public`. This would + // provide a way for downstream code (within core and plugins) to retrieve info about the executed plugins + @TestVisible + private Map> getExecutedApexPlugins() { + return this.methodToExecutedApexPlugins; + } + + @TestVisible + private Map> getExecutedFlowPlugins() { + return this.methodToExecutedFlowPlugins; + } } diff --git a/nebula-logger/core/main/log-management/classes/LogEntryEventHandler.cls b/nebula-logger/core/main/log-management/classes/LogEntryEventHandler.cls index c15d7ab5e..b4054d678 100644 --- a/nebula-logger/core/main/log-management/classes/LogEntryEventHandler.cls +++ b/nebula-logger/core/main/log-management/classes/LogEntryEventHandler.cls @@ -7,8 +7,10 @@ * @group Log Management * @description Processes `LogEntryEvent__e` platform events and normalizes the data into `Log__c` and `LogEntry__c` records */ -@SuppressWarnings('PMD.ApexCRUDViolation, PMD.AvoidDebugStatements, PMD.CyclomaticComplexity, PMD.CognitiveComplexity') +@SuppressWarnings('PMD.ApexCrudViolation, PMD.AvoidDebugStatements, PMD.CognitiveComplexity, PMD.CyclomaticComplexity') public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { + @TestVisible + private static final String DEFAULT_STORAGE_LOCATION_NAME = 'CUSTOM_OBJECTS'; private static final Database.DmlOptions DML_OPTIONS = createDmlOptions(); private static final Map TRANSACTION_ID_TO_LOG = new Map(); @TestVisible @@ -22,32 +24,29 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { private Set tagNames = new Set(); /** - * @description Returns SObject Type that the handler is responsible for processing - * @return The instance of `SObjectType` + * @description Default constructor, used by the trigger `LogEntryEvent.trigger` */ - public override SObjectType getSObjectType() { - return Schema.LogEntryEvent__e.SObjectType; + public LogEntryEventHandler() { + super(); } /** - * @description method so it can be called from Logger - this is unique to LogEntryEventHandler - * @param triggerNew list of new LogEntryEvent events + * @description Returns SObject Type that the handler is responsible for processing + * @return The instance of `SObjectType` */ - public override void executeBeforeInsert(List triggerNew) { - this.logEntryEvents = (List) triggerNew; + public override SObjectType getSObjectType() { + return Schema.LogEntryEvent__e.SObjectType; } - /** - * @description Public method so it can be called from Logger - this is unique to LogEntryEventHandler - * @param triggerNew list of new LogEntryEvent events - */ - public override void executeAfterInsert(List triggerNew) { + protected override void executeAfterInsert(List triggerNew) { this.logEntryEvents = this.filterLogEntryEventsToSave((List) triggerNew); - this.upsertLogs(); - this.insertLogEntries(); - this.appendRuleBasedTags(); - this.insertLogEntryTags(); + if (this.logEntryEvents.isEmpty() == false) { + this.upsertLogs(); + this.insertLogEntries(); + this.appendRuleBasedTags(); + this.insertLogEntryTags(); + } } private List filterLogEntryEventsToSave(List newLogEntryEvents) { @@ -55,7 +54,7 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { for (LogEntryEvent__e logEntryEvent : newLogEntryEvents) { User loggingUser = new User(Id = logEntryEvent.LoggedById__c, ProfileId = logEntryEvent.ProfileId__c); LoggerSettings__c loggingUserSettings = Logger.getUserSettings(loggingUser); - if (loggingUserSettings.IsPlatformEventStorageEnabled__c == true) { + if (loggingUserSettings.DefaultPlatformEventStorageLocation__c == DEFAULT_STORAGE_LOCATION_NAME) { logEntryEventsToSave.add(logEntryEvent); } } @@ -142,10 +141,12 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { TRANSACTION_ID_TO_LOG.put(log.TransactionId__c, log); } - List upsertResults = Database.upsert(TRANSACTION_ID_TO_LOG.values(), Schema.Log__c.TransactionId__c, false); - LoggerEmailUtils.sendErrorEmail(Schema.Log__c.SObjectType, upsertResults); + List upsertResults = LoggerDataStore.getDatabase() + .upsertRecords(TRANSACTION_ID_TO_LOG.values(), Schema.Log__c.TransactionId__c, System.Test.isRunningTest()); + LoggerEmailSender.sendErrorEmail(Schema.Log__c.SObjectType, upsertResults); // If no recent logs have the details, and there is not another instance of the job in progress, then start a new one + // TODO this probably should be moved to LogHandler instead of here if (LoggerParameter.CALL_STATUS_API == true && recentLogWithApiReleaseDetails == null && getCountOfOpenJobs() == 0) { setStatusApiDetails(); } @@ -160,6 +161,7 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { LogEntry__c logEntry = new LogEntry__c( ComponentType__c = logEntryEvent.ComponentType__c, + DatabaseResultCollectionSize__c = logEntryEvent.DatabaseResultCollectionSize__c, DatabaseResultCollectionType__c = logEntryEvent.DatabaseResultCollectionType__c, DatabaseResultJson__c = logEntryEvent.DatabaseResultJson__c, DatabaseResultType__c = logEntryEvent.DatabaseResultType__c, @@ -168,6 +170,16 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { ExceptionMessage__c = logEntryEvent.ExceptionMessage__c, ExceptionStackTrace__c = logEntryEvent.ExceptionStackTrace__c, ExceptionType__c = logEntryEvent.ExceptionType__c, + HttpRequestBody__c = logEntryEvent.HttpRequestBody__c, + HttpRequestBodyMasked__c = logEntryEvent.HttpRequestBodyMasked__c, + HttpRequestCompressed__c = logEntryEvent.HttpRequestCompressed__c, + HttpRequestEndpoint__c = logEntryEvent.HttpRequestEndpoint__c, + HttpRequestMethod__c = logEntryEvent.HttpRequestMethod__c, + HttpResponseBody__c = logEntryEvent.HttpResponseBody__c, + HttpResponseBodyMasked__c = logEntryEvent.HttpResponseBodyMasked__c, + HttpResponseHeaderKeys__c = logEntryEvent.HttpResponseHeaderKeys__c, + HttpResponseStatus__c = logEntryEvent.HttpResponseStatus__c, + HttpResponseStatusCode__c = logEntryEvent.HttpResponseStatusCode__c, LimitsAggregateQueriesMax__c = logEntryEvent.LimitsAggregateQueriesMax__c, LimitsAggregateQueriesUsed__c = logEntryEvent.LimitsAggregateQueriesUsed__c, LimitsAsyncCallsMax__c = logEntryEvent.LimitsAsyncCallsMax__c, @@ -209,6 +221,7 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { Name = null, // Salesforce will auto-set the record ID as the name when null OriginLocation__c = logEntryEvent.OriginLocation__c, OriginType__c = logEntryEvent.OriginType__c, + RecordCollectionSize__c = logEntryEvent.RecordCollectionSize__c, RecordCollectionType__c = logEntryEvent.RecordCollectionType__c, RecordId__c = logEntryEvent.RecordId__c, RecordJson__c = logEntryEvent.RecordJson__c, @@ -233,8 +246,8 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { } } - List saveResults = Database.insert(this.logEntries, DML_OPTIONS); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResults); + List saveResults = LoggerDataStore.getDatabase().insertRecords(this.logEntries, DML_OPTIONS); + LoggerEmailSender.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResults); } private void appendRuleBasedTags() { @@ -295,8 +308,8 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { } } - List saveResults = Database.insert(new List(tagAssignments), DML_OPTIONS); - LoggerEmailUtils.sendErrorEmail(tagAssignmentSObjectType, saveResults); + List saveResults = LoggerDataStore.getDatabase().insertRecords(new List(tagAssignments), DML_OPTIONS); + LoggerEmailSender.sendErrorEmail(tagAssignmentSObjectType, saveResults); } private Map getTagNameToId(Schema.SObjectType tagSObjectType) { @@ -324,8 +337,8 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { } if (!missingTagsToCreate.isEmpty()) { - List saveResults = Database.insert(missingTagsToCreate, DML_OPTIONS); - LoggerEmailUtils.sendErrorEmail(tagSObjectType, saveResults); + List saveResults = LoggerDataStore.getDatabase().insertRecords(missingTagsToCreate, DML_OPTIONS); + LoggerEmailSender.sendErrorEmail(tagSObjectType, saveResults); for (SObject tag : missingTagsToCreate) { missingTagNameToId.put((String) tag.get('Name'), (Id) tag.get('Id')); } @@ -338,7 +351,7 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { private static Database.DmlOptions createDmlOptions() { Database.DmlOptions dmlOptions = new Database.DmlOptions(); dmlOptions.AllowFieldTruncation = true; - dmlOptions.OptAllOrNone = false; + dmlOptions.OptAllOrNone = System.Test.isRunningTest(); return dmlOptions; } @@ -427,7 +440,9 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { @future(callout=true) private static void setStatusApiDetails() { - System.debug(LoggingLevel.DEBUG, 'Logger - Calling Status API for org details'); + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.debug('Logger - Calling Status API for org details'); + } Organization organization = [SELECT InstanceName FROM Organization]; String statusApiEndpoint = 'https://api.status.salesforce.com/v1/instances/' + organization.InstanceName + '/status'; @@ -450,7 +465,9 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { } StatusApiResponse statusApiResponse = (StatusApiResponse) JSON.deserialize(response.getBody(), StatusApiResponse.class); - System.debug(LoggingLevel.DEBUG, 'Logger - Status API response: ' + statusApiResponse); + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.debug('Logger - Status API response: ' + statusApiResponse); + } List logsToUpdate = new List(); for (Log__c log : [ @@ -465,8 +482,13 @@ public without sharing class LogEntryEventHandler extends LoggerSObjectHandler { logsToUpdate.add(log); } - System.debug(LoggingLevel.DEBUG, 'Logger - logs to update: ' + logsToUpdate); - update logsToUpdate; + + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.debug('Logger - logs to update: ' + logsToUpdate); + } + + LoggerDataStore.getDatabase().updateRecords(logsToUpdate); + Logger.saveLog(); } // Private class for handling the response from api.status.salesforce.com diff --git a/nebula-logger/core/main/log-management/classes/LogEntryHandler.cls b/nebula-logger/core/main/log-management/classes/LogEntryHandler.cls index 71baae386..d31a8897b 100644 --- a/nebula-logger/core/main/log-management/classes/LogEntryHandler.cls +++ b/nebula-logger/core/main/log-management/classes/LogEntryHandler.cls @@ -9,6 +9,8 @@ */ @SuppressWarnings('PMD.ApexCrudViolation, PMD.CognitiveComplexity, PMD.CyclomaticComplexity, PMD.NcssMethodCount, PMD.StdCyclomaticComplexity') public without sharing class LogEntryHandler extends LoggerSObjectHandler { + private static final Map SOBJECT_TYPE_TO_DISPLAY_FIELD_NAME = new Map(); + @TestVisible private List logEntries; @@ -94,10 +96,6 @@ public without sharing class LogEntryHandler extends LoggerSObjectHandler { innerApexClassName = innerApexClassName.substringAfter(topLevelApexClassName + '.'); } - // TODO confirm if there any issues with automated users (e.g., PackagePushJob uses a not-quite-a-user) - // logEntry.ApexClassCreatedById__c = topLevelApexClass.CreatedById; - // logEntry.ApexClassLastModifiedById__c = apexClass.LastModifiedById; - logEntry.ApexClassApiVersion__c = 'v' + topLevelApexClass.ApiVersion; logEntry.ApexClassCreatedDate__c = topLevelApexClass.CreatedDate; logEntry.ApexClassId__c = topLevelApexClass.Id; @@ -238,8 +236,9 @@ public without sharing class LogEntryHandler extends LoggerSObjectHandler { for (Schema.SObjectType sobjectType : sobjectTypeToRecords.keySet()) { // Stop if we've used up all the queries if (Limits.getQueries() == Limits.getLimitQueries()) { - return; + break; } + List sobjectTypeRecordIds = sobjectTypeToRecords.get(sobjectType); String sobjectDisplayFieldName = this.getDisplayFieldApiName(sobjectType); @@ -277,9 +276,15 @@ public without sharing class LogEntryHandler extends LoggerSObjectHandler { } private String getDisplayFieldApiName(Schema.SObjectType sobjectType) { + if (SOBJECT_TYPE_TO_DISPLAY_FIELD_NAME.containsKey(sobjectType) == true) { + return SOBJECT_TYPE_TO_DISPLAY_FIELD_NAME.get(sobjectType); + } + // Use username instead of name for user if (sobjectType == Schema.User.SObjectType) { - return Schema.User.Username.getDescribe().getName(); + String userFieldName = Schema.User.Username.getDescribe().getName(); + SOBJECT_TYPE_TO_DISPLAY_FIELD_NAME.put(Schema.User.SObjectType, userFieldName); + return userFieldName; } // There are several commonly used names for the display field name - typically, Name @@ -305,6 +310,7 @@ public without sharing class LogEntryHandler extends LoggerSObjectHandler { }; String displayFieldApiName; + List fallbackFieldApiNames = new List(); for (String fieldName : educatedGuesses) { Schema.SObjectField field = sobjectType.getDescribe().fields.getMap().get(fieldName); @@ -314,19 +320,23 @@ public without sharing class LogEntryHandler extends LoggerSObjectHandler { Schema.DescribeFieldResult fieldDescribe = field.getDescribe(); - if (fieldDescribe.isNameField()) { + if (fieldDescribe.isNameField() == true) { displayFieldApiName = fieldDescribe.getName(); break; + } else { + fallbackFieldApiNames.add(fieldDescribe.getName()); } } - return displayFieldApiName; - } - - private static String getNamespacePrefix() { - String className = LogEntryHandler.class.getName(); - String namespacePrefix = className.contains('.') ? className.substringBefore('.') : ''; + // Some objects don't have a proper 'display field', but they do have a reasonable alternative field to use instead. + // For example, EmailMessage doesn't have a true display field, but it has a 'Subject' field. If one (and only one) + // of the educated guess-fields exists on the object - and the object does not have a display field - then use the + // educated guess-field as the fallback for the display field. + if (String.isBlank(displayFieldApiName) == true && fallbackFieldApiNames.size() == 1) { + displayFieldApiName = fallbackFieldApiNames.get(0); + } - return namespacePrefix; + SOBJECT_TYPE_TO_DISPLAY_FIELD_NAME.put(sobjectType, displayFieldApiName); + return displayFieldApiName; } } diff --git a/nebula-logger/core/main/log-management/classes/LogHandler.cls b/nebula-logger/core/main/log-management/classes/LogHandler.cls index 3f0361185..bd7b7ab2e 100644 --- a/nebula-logger/core/main/log-management/classes/LogHandler.cls +++ b/nebula-logger/core/main/log-management/classes/LogHandler.cls @@ -9,10 +9,8 @@ */ @SuppressWarnings('PMD.ApexCrudViolation, PMD.CognitiveComplexity, PMD.CyclomaticComplexity') public without sharing class LogHandler extends LoggerSObjectHandler { - @TestVisible - private static final Map LOG_STATUS_NAME_TO_STATUS = loadActiveLogStatuses(); - @TestVisible - private static final Map SCENARIO_TO_MOCK_SCENARIO_RULE = new Map(); + private static final Map MOCK_LOG_STATUS_TO_STATUS = new Map(); + private static final Map MOCK_SCENARIO_TO_SCENARIO_RULE = new Map(); @TestVisible private List logs; @@ -34,7 +32,7 @@ public without sharing class LogHandler extends LoggerSObjectHandler { // The log OwnerId field should support being manually changed, so only auto-set it on insert this.setOwnerId(); // The log retention date field should support being manually changed, so only auto-set it on insert - this.setLogRetentionDate(); + this.setLogRetentionDetails(); } protected override void executeBeforeUpdate(Map triggerNewMap, Map triggerOldMap) { @@ -53,9 +51,10 @@ public without sharing class LogHandler extends LoggerSObjectHandler { } private void setClosedStatusFields() { + Map logStatusNameToStatus = loadActiveLogStatuses(); for (Log__c log : this.logs) { // Determine if the status is considered closed (based on `LogStatus__mdt` custom metadata) - LogStatus__mdt logStatus = LOG_STATUS_NAME_TO_STATUS.get(log.Status__c); + LogStatus__mdt logStatus = logStatusNameToStatus.get(log.Status__c); if (logStatus != null) { log.IsClosed__c = logStatus.IsClosed__c; log.IsResolved__c = logStatus.IsResolved__c; @@ -106,7 +105,7 @@ public without sharing class LogHandler extends LoggerSObjectHandler { } } - private void setLogRetentionDate() { + private void setLogRetentionDetails() { Map scenarioToScenarioRule = queryScenarioRules(this.logs); for (Log__c log : this.logs) { // If the retention date has already been populated, leave it as-is @@ -119,17 +118,16 @@ public without sharing class LogHandler extends LoggerSObjectHandler { // Load the configured scenario rule (if one exists) LogScenarioRule__mdt matchingScenarioRule = scenarioToScenarioRule.get(log.Scenario__c); - if (System.Test.isRunningTest() == true) { - matchingScenarioRule = SCENARIO_TO_MOCK_SCENARIO_RULE.get(log.Scenario__c); - } Integer daysToRetainLog = Integer.valueOf( matchingScenarioRule != null ? matchingScenarioRule.NumberOfDaysToRetainLogs__c : loggingUserSettings.DefaultNumberOfDaysToRetainLogs__c ); + // TODO Add new field to LogScenarioRule__mdt for setting the default log purge action (same concept as matchingScenarioRule.NumberOfDaysToRetainLogs__c above) // When daysToRetainLog is null, assume that the log should be kept forever, // and set the retention date to null so that LogBatchPurger filters out/ignores the record log.LogRetentionDate__c = daysToRetainLog == null ? null : System.today().addDays(daysToRetainLog); + log.LogPurgeAction__c = loggingUserSettings.DefaultLogPurgeAction__c; } } @@ -179,7 +177,7 @@ public without sharing class LogHandler extends LoggerSObjectHandler { ); logShares.add(logShare); } - Database.insert(logShares, false); + LoggerDataStore.getDatabase().insertRecords(logShares, false); } private static LoggerSettings__c getLoggingUserSettings(Log__c log) { @@ -194,6 +192,12 @@ public without sharing class LogHandler extends LoggerSObjectHandler { logStatusNameToStatus.put(logStatus.MasterLabel, logStatus); } } + + if (System.Test.isRunningTest() == true) { + logStatusNameToStatus.clear(); + logStatusNameToStatus.putAll(MOCK_LOG_STATUS_TO_STATUS); + } + return logStatusNameToStatus; } @@ -229,11 +233,22 @@ public without sharing class LogHandler extends LoggerSObjectHandler { ]) { scenarioToScenarioRule.put(scenarioRule.Scenario__c, scenarioRule); } + + if (System.Test.isRunningTest() == true) { + scenarioToScenarioRule.clear(); + scenarioToScenarioRule.putAll(MOCK_SCENARIO_TO_SCENARIO_RULE); + } + return scenarioToScenarioRule; } + @TestVisible + private static void setMockLogStatus(LogStatus__mdt logStatus) { + MOCK_LOG_STATUS_TO_STATUS.put(logStatus.MasterLabel, logStatus); + } + @TestVisible private static void setMockScenarioRule(LogScenarioRule__mdt scenarioRule) { - SCENARIO_TO_MOCK_SCENARIO_RULE.put(scenarioRule.Scenario__c, scenarioRule); + MOCK_SCENARIO_TO_SCENARIO_RULE.put(scenarioRule.Scenario__c, scenarioRule); } } diff --git a/nebula-logger/core/main/log-management/classes/LogMassDeleteExtension.cls b/nebula-logger/core/main/log-management/classes/LogMassDeleteExtension.cls index 8bc7b50d9..feecb13c4 100644 --- a/nebula-logger/core/main/log-management/classes/LogMassDeleteExtension.cls +++ b/nebula-logger/core/main/log-management/classes/LogMassDeleteExtension.cls @@ -55,7 +55,7 @@ public with sharing class LogMassDeleteExtension { * @return The PageReference of the previous page (based on `controller.cancel()`) */ public PageReference deleteSelectedLogs() { - delete getDeletableLogs(); + LoggerDataStore.getDatabase().deleteRecords(getDeletableLogs()); // The controller's method cancel() just returns the user to the previous page - it doesn't rollback any DML statements (like the delete above) return this.controller.cancel(); diff --git a/nebula-logger/core/main/log-management/classes/LoggerBatchableContext.cls b/nebula-logger/core/main/log-management/classes/LoggerBatchableContext.cls new file mode 100644 index 000000000..14815b66d --- /dev/null +++ b/nebula-logger/core/main/log-management/classes/LoggerBatchableContext.cls @@ -0,0 +1,30 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Log Management + * @description Class used by the logging system for batch contextual details + * @see LogBatchPurger + * @see LoggerPlugin + */ +@SuppressWarnings('PMD.ApexDoc') +public without sharing class LoggerBatchableContext { + @AuraEnabled + public Database.BatchableContext batchableContext; + public transient Schema.SObjectType sobjectType; + @AuraEnabled + public String sobjectTypeName; + + /** + * @description Constructor used to set the 2 properties + * @param batchableContext The instance of `Database.BatchableContextbatchableContext`, provided by the platform at run-time + * @param sobjectType The `SObjectType` that will be queried & purged + */ + public LoggerBatchableContext(Database.BatchableContext batchableContext, Schema.SObjectType sobjectType) { + this.batchableContext = batchableContext; + this.sobjectType = sobjectType; + this.sobjectTypeName = sobjectType?.getDescribe().getName(); + } +} diff --git a/nebula-logger/core/main/log-management/classes/LoggerSObjectHandler.cls-meta.xml b/nebula-logger/core/main/log-management/classes/LoggerBatchableContext.cls-meta.xml similarity index 100% rename from nebula-logger/core/main/log-management/classes/LoggerSObjectHandler.cls-meta.xml rename to nebula-logger/core/main/log-management/classes/LoggerBatchableContext.cls-meta.xml diff --git a/nebula-logger/core/main/configuration/classes/LoggerEmailUtils.cls b/nebula-logger/core/main/log-management/classes/LoggerEmailSender.cls similarity index 83% rename from nebula-logger/core/main/configuration/classes/LoggerEmailUtils.cls rename to nebula-logger/core/main/log-management/classes/LoggerEmailSender.cls index 7770cab70..916e11692 100644 --- a/nebula-logger/core/main/configuration/classes/LoggerEmailUtils.cls +++ b/nebula-logger/core/main/log-management/classes/LoggerEmailSender.cls @@ -4,11 +4,16 @@ //------------------------------------------------------------------------------------------------// /** - * @group Configuration + * @group Log Management * @description Builds and sends email notifications when internal exceptions occur within the logging system */ @SuppressWarnings('PMD.PropertyNamingConventions') -public without sharing class LoggerEmailUtils { +public without sharing class LoggerEmailSender { + @TestVisible + private static final List MOCK_NOTIFICATIONS = new List(); + @TestVisible + private static final List SENT_EMAILS = new List(); + @TestVisible private static final List CACHED_APEX_ERROR_RECIPIENTS { get { @@ -20,17 +25,16 @@ public without sharing class LoggerEmailUtils { set; } + @TestVisible private static Boolean IS_EMAIL_DELIVERABILITY_ENABLED { get { if (IS_EMAIL_DELIVERABILITY_ENABLED == null) { try { - Messaging.reserveSingleEmailCapacity(1); - Messaging.reserveMassEmailCapacity(1); + System.Messaging.reserveSingleEmailCapacity(1); + System.Messaging.reserveMassEmailCapacity(1); IS_EMAIL_DELIVERABILITY_ENABLED = true; - return IS_EMAIL_DELIVERABILITY_ENABLED; } catch (System.NoAccessException e) { IS_EMAIL_DELIVERABILITY_ENABLED = false; - return IS_EMAIL_DELIVERABILITY_ENABLED; } } return IS_EMAIL_DELIVERABILITY_ENABLED; @@ -38,17 +42,6 @@ public without sharing class LoggerEmailUtils { set; } - @TestVisible - private static final List SENT_EMAILS { - get { - if (SENT_EMAILS == null) { - SENT_EMAILS = new List(); - } - return SENT_EMAILS; - } - set; - } - /** * @description Sends an error email notification to the org's list of Apex Exception Email recipients, * configured under Setup --> Email --> Apex Exception Email @@ -88,7 +81,9 @@ public without sharing class LoggerEmailUtils { } if (CACHED_APEX_ERROR_RECIPIENTS.isEmpty() == true) { - System.debug(LoggingLevel.INFO, 'Logger - no Apex email recipients configured, skipping sending email'); + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.info('Logger - no Apex email recipients configured, skipping sending email'); + } return; } @@ -129,10 +124,13 @@ public without sharing class LoggerEmailUtils { List messages = new List{ message }; List emailResults = Messaging.sendEmail(messages); SENT_EMAILS.add(message); - if (emailResults[0].success == true) { - System.debug(LoggingLevel.INFO, 'Logger - The email was sent successfully'); + + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == false) { + return; + } else if (emailResults.get(0).success == true) { + Logger.info('Logger - The email was sent successfully'); } else { - System.debug(LoggingLevel.INFO, 'Logger - The email failed to send: ' + emailResults[0].errors[0].message); + Logger.info('Logger - The email failed to send: ' + emailResults.get(0).errors.get(0).message); } } } @@ -159,10 +157,17 @@ public without sharing class LoggerEmailUtils { @SuppressWarnings('PMD.ApexCRUDViolation') private static List queryApexErrrorRecipients() { List apexErrrorRecipients = new List(); - for (ApexEmailNotification notification : [SELECT Email, UserId FROM ApexEmailNotification WHERE Email != NULL OR User.IsActive = TRUE]) { + List notifications = [SELECT Email, UserId FROM ApexEmailNotification WHERE Email != NULL OR User.IsActive = TRUE]; + + if (System.Test.isRunningTest() == true) { + notifications.clear(); + notifications.addAll(MOCK_NOTIFICATIONS); + } + + for (ApexEmailNotification notification : notifications) { if (notification.UserId != null) { apexErrrorRecipients.add(notification.UserId); - } else { + } else if (String.isNotBlank(notification.Email) == true) { apexErrrorRecipients.addAll(notification.Email.split(';')); } } diff --git a/nebula-logger/core/main/plugin-framework/classes/LoggerSObjectHandlerPlugin.cls-meta.xml b/nebula-logger/core/main/log-management/classes/LoggerEmailSender.cls-meta.xml similarity index 100% rename from nebula-logger/core/main/plugin-framework/classes/LoggerSObjectHandlerPlugin.cls-meta.xml rename to nebula-logger/core/main/log-management/classes/LoggerEmailSender.cls-meta.xml diff --git a/nebula-logger/core/main/log-management/classes/LoggerSObjectHandler.cls b/nebula-logger/core/main/log-management/classes/LoggerSObjectHandler.cls deleted file mode 100644 index 28c303715..000000000 --- a/nebula-logger/core/main/log-management/classes/LoggerSObjectHandler.cls +++ /dev/null @@ -1,182 +0,0 @@ -//------------------------------------------------------------------------------------------------// -// This file is part of the Nebula Logger project, released under the MIT License. // -// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // -//------------------------------------------------------------------------------------------------// - -/** - * @group Log Management - * @description Abstract class used by trigger handlers for shared logic - */ -@SuppressWarnings('PMD.ApexCRUDViolation, PMD.EmptyStatementBlock') -public without sharing abstract class LoggerSObjectHandler { - private static Map> pluginsBySObjectType = new Map>(); - - @TestVisible - private TriggerOperation triggerOperationType; - @TestVisible - private List triggerNew; - @TestVisible - private Map triggerNewMap; - @TestVisible - private List triggerOld; - @TestVisible - private Map triggerOldMap; - - private List plugins; - - static { - // Query the LoggerPlugin__mdt CMDT object so the records are sorted correctly - List plugins = [ - SELECT SObjectType__r.QualifiedApiName, PluginType__c, PluginApiName__c - FROM LoggerPlugin__mdt - WHERE IsEnabled__c = TRUE - ORDER BY ExecutionOrder__c NULLS LAST, DeveloperName - ]; - - for (LoggerPlugin__mdt plugin : plugins) { - // CMDT entity-definition relationship fields are weird, so skip some headaches by copying the Qualified API name - plugin.SObjectType__c = plugin.SObjectType__r.QualifiedApiName; - - // Schema.getGlobalDescribe() is the worst, so don't use it - SObjectType sobjectType = ((SObject) Type.forName(plugin.SObjectType__c).newInstance()).getSObjectType(); - - List sobjectTypePlugins = new List(); - if (pluginsBySObjectType.containsKey(sobjectType) == true) { - sobjectTypePlugins = pluginsBySObjectType.get(sobjectType); - } - sobjectTypePlugins.add(plugin); - pluginsBySObjectType.put(sobjectType, sobjectTypePlugins); - } - - if (System.Test.isRunningTest() == true) { - // Tests shouldn't rely on the actual CMDT rules in the org - // Clear the org's loaded records during tests, and mock via setMockConfiguration() - pluginsBySObjectType.clear(); - } - } - - @TestVisible - private static void setMockPlugin(Schema.SObjectType sobjectType, LoggerPlugin__mdt plugin) { - List plugins = pluginsBySObjectType.get(sobjectType); - if (plugins == null) { - plugins = new List(); - } - plugins.add(plugin); - pluginsBySObjectType.put(sobjectType, plugins); - } - - /** - * @description Default constructor - */ - public LoggerSObjectHandler() { - this.triggerOperationType = Trigger.operationType; - this.triggerNew = Trigger.new; - this.triggerNewMap = Trigger.newMap; - this.triggerOld = Trigger.old; - this.triggerOldMap = Trigger.oldMap; - this.plugins = pluginsBySObjectType.get(this.getSObjectType()); - } - - /** - * @description Returns the SObject Type that the handler is responsible for processing - * @return The instance of `SObjectType` - */ - public abstract SObjectType getSObjectType(); - - /** - * @description Runs the handler class's logic - */ - public void execute() { - switch on this.triggerOperationType { - when BEFORE_INSERT { - this.executeBeforeInsert(this.triggerNew); - } - when BEFORE_UPDATE { - this.executeBeforeUpdate(this.triggerNewMap, this.triggerOldMap); - } - when BEFORE_DELETE { - this.executeBeforeDelete(this.triggerNewMap); - } - when AFTER_INSERT { - // Platform Events don't have an ID field, thus Trigger.newMap doesn't work for LogEntryEvent__e - // For custom objects, Map is more convenient since it provides both the keys & values - // 2 AFTER_INSERT methods are used here in the framework, with the expectation that only 1 will be implemented per handler class - this.executeAfterInsert(this.triggerNew); - this.executeAfterInsert(this.triggerNewMap); - } - when AFTER_UPDATE { - this.executeAfterUpdate(this.triggerNewMap, this.triggerOldMap); - } - when AFTER_DELETE { - this.executeAfterDelete(this.triggerNewMap); - } - when AFTER_UNDELETE { - this.executeAfterUndelete(this.triggerNewMap); - } - } - - this.executePlugins(); - } - - protected virtual void executeBeforeInsert(List triggerNew) { - } - - protected virtual void executeBeforeUpdate(Map triggerNewMap, Map triggerOldMap) { - } - - protected virtual void executeBeforeDelete(Map triggerNewMap) { - } - - // executeAfterInsert(List triggerNew) { - } - - protected virtual void executeAfterInsert(Map triggerNewMap) { - } - - protected virtual void executeAfterUpdate(Map triggerNewMap, Map triggerOldMap) { - } - - protected virtual void executeAfterDelete(Map triggerNewMap) { - } - - protected virtual void executeAfterUndelete(Map triggerNewMap) { - } - - private void executePlugins() { - if (this.plugins == null || this.plugins.isEmpty() == true) { - return; - } - - for (LoggerPlugin__mdt pluginConfiguration : this.plugins) { - switch on pluginConfiguration.PluginType__c { - when 'Apex' { - this.executeApexPlugin(pluginConfiguration.PluginApiName__c); - } - when 'Flow' { - this.executeFlowPlugin(pluginConfiguration.PluginApiName__c); - } - } - } - } - - private void executeApexPlugin(String apexClassName) { - LoggerSObjectHandlerPlugin apexPlugin = (LoggerSObjectHandlerPlugin) Type.forName(apexClassName).newInstance(); - apexPlugin.execute(this.triggerOperationType, this.triggerNew, this.triggerNewMap, this.triggerOld, this.triggerOldMap); - } - - private void executeFlowPlugin(String flowApiName) { - Map flowInputs = new Map(); - flowInputs.put('triggerOperationType', this.triggerOperationType?.name()); - flowInputs.put('triggerNew', this.triggerNew); - flowInputs.put('triggerOld', this.triggerOld); - - Flow.Interview flowPlugin = Flow.Interview.createInterview(flowApiName, flowInputs); - flowPlugin.start(); - - List updatedTriggerNew = (List) flowPlugin.getVariableValue('updatedTriggerNew'); - if (updatedTriggerNew != null && updatedTriggerNew.size() == this.triggerNew.size()) { - this.triggerNew = updatedTriggerNew; - } - } -} diff --git a/nebula-logger/core/main/log-management/classes/LoggerSObjectMetadata.cls b/nebula-logger/core/main/log-management/classes/LoggerSObjectMetadata.cls index 26dc92c89..785da5f2b 100644 --- a/nebula-logger/core/main/log-management/classes/LoggerSObjectMetadata.cls +++ b/nebula-logger/core/main/log-management/classes/LoggerSObjectMetadata.cls @@ -9,24 +9,27 @@ */ public without sharing class LoggerSObjectMetadata { /** - * @description Provides schema details about the the platform event object `LogEntryEvent__e` - * @return An instance of `LoggerSObjectMetadata.SObjectSchema` for the platform event `LogEntryEvent__e` + * @description Provides schema details about the specified `SObjectType` + * @param sobjectApiName The API name of the `SObjectType` to convert to an instance of `LoggerSObjectMetadata.SObjectSchema` + * @return An instance of `LoggerSObjectMetadata.SObjectSchema` for the specified `SObjectType` */ @AuraEnabled(cacheable=true) - public static SObjectSchema getLogEntryEventSchema() { - return buildSObjectSchema(Schema.LogEntryEvent__e.SObjectType.getDescribe()); + public static SObjectSchema getSchemaForName(String sobjectApiName) { + String namespacePrefix = getNamespacePrefix(); + String qualifiedApiName = String.isBlank(namespacePrefix) == true ? sobjectApiName : namespacePrefix + '__' + sobjectApiName; + Type apexType = Type.forName(qualifiedApiName) != null ? Type.forName(qualifiedApiName) : Type.forName(sobjectApiName); + Schema.SObjectType sobjectType = ((SObject) apexType.newInstance()).getSObjectType(); + return getSchema(sobjectType); } /** - * @description Provides schema details about the the custom settings object `LoggerSettings__c` - * @return An instance of `LoggerSObjectMetadata.SObjectSchema` for the platform event `LoggerSettings__c` + * @description Provides schema details about the specified `SObjectType` + * @param sobjectType The instance of `SObjectType` to convert to an instance of `LoggerSObjectMetadata.SObjectSchema` + * @return An instance of `LoggerSObjectMetadata.SObjectSchema` for the specified `SObjectType` */ - @AuraEnabled(cacheable=true) - public static SObjectSchema getLoggerSettingsSchema() { - return buildSObjectSchema(Schema.LoggerSettings__c.SObjectType.getDescribe()); - } + public static SObjectSchema getSchema(Schema.SObjectType sobjectType) { + Schema.DescribeSObjectResult describe = sobjectType.getDescribe(); - private static SObjectSchema buildSObjectSchema(Schema.DescribeSObjectResult describe) { SObjectSchema schema = new SObjectSchema(); schema.apiName = describe.getName(); schema.localApiName = describe.getLocalName(); @@ -53,6 +56,13 @@ public without sharing class LoggerSObjectMetadata { return schema; } + private static String getNamespacePrefix() { + String className = LoggerSObjectMetadata.class.getName(); + String namespacePrefix = className.contains('.') ? className.substringBefore('.') : ''; + + return namespacePrefix; + } + /** * @description Inner class for `SObject` details to LWCs, using `@AuraEnabled` properties */ diff --git a/nebula-logger/core/main/log-management/classes/LoggerSettingsController.cls b/nebula-logger/core/main/log-management/classes/LoggerSettingsController.cls index 24ada1915..5f4545a75 100644 --- a/nebula-logger/core/main/log-management/classes/LoggerSettingsController.cls +++ b/nebula-logger/core/main/log-management/classes/LoggerSettingsController.cls @@ -9,6 +9,17 @@ */ @SuppressWarnings('PMD.ApexCRUDViolation, PMD.CyclomaticComplexity, PMD.ExcessivePublicCount') public without sharing class LoggerSettingsController { + @TestVisible + private static final String CUSTOM_SAVE_METHOD_PREFIX = 'CustomSaveMethod'; + @TestVisible + private static final String CUSTOM_OBJECTS_STORAGE_LOCATION = 'CUSTOM_OBJECTS'; + @TestVisible + private static final String CUSTOM_LOG_PURGE_ACTION_PREFIX = 'CustomLogPurgeAction'; + @TestVisible + private static final String CUSTOM_STORAGE_LOCATION_PREFIX = 'CustomStorageLocation'; + @TestVisible + private static final String DELETE_LOG_PURGE_ACTION = 'Delete'; + // Data methods /** * @description Indicates if the current user has access to modify `LoggerSettings__c` records, @@ -29,6 +40,8 @@ public without sharing class LoggerSettingsController { public static LoggerSettingsPicklistOptions getPicklistOptions() { LoggerSettingsPicklistOptions loggerSettingsPicklistOptions = new LoggerSettingsPicklistOptions(); loggerSettingsPicklistOptions.loggingLevelOptions = getLoggingLevelOptions(); + loggerSettingsPicklistOptions.platformEventStorageLocationOptions = getPlatformEventStorageLocationOptions(); + loggerSettingsPicklistOptions.purgeActionOptions = getPurgeActionOptions(); loggerSettingsPicklistOptions.saveMethodOptions = getSaveMethodOptions(); loggerSettingsPicklistOptions.setupOwnerTypeOptions = getSetupOwnerTypeOptions(); loggerSettingsPicklistOptions.shareAccessLevelOptions = getShareAccessLevelOptions(); @@ -83,7 +96,7 @@ public without sharing class LoggerSettingsController { public static void saveRecord(LoggerSettings__c settingsRecord) { try { if (canUserModifyLoggerSettings() == true) { - upsert settingsRecord; + LoggerDataStore.getDatabase().upsertRecord(settingsRecord, Schema.LoggerSettings__c.Id); } } catch (Exception ex) { throw createAuraHandledException(ex); @@ -98,7 +111,7 @@ public without sharing class LoggerSettingsController { public static void deleteRecord(LoggerSettings__c settingsRecord) { try { if (canUserModifyLoggerSettings() == true) { - delete settingsRecord; + LoggerDataStore.getDatabase().deleteRecord(settingsRecord); } } catch (Exception ex) { throw createAuraHandledException(ex); @@ -187,12 +200,57 @@ public without sharing class LoggerSettingsController { return picklistOptions; } - private static List getSaveMethodOptions() { + private static List getPlatformEventStorageLocationOptions() { + List storageLocationNames = new List{ CUSTOM_OBJECTS_STORAGE_LOCATION }; + for (LoggerParameter__mdt matchingAdditionalParameter : LoggerParameter.matchOnPrefix(CUSTOM_STORAGE_LOCATION_PREFIX)) { + storageLocationNames.add(matchingAdditionalParameter.Value__c); + } + storageLocationNames.sort(); + + List picklistOptions = initializePicklistOptions(); + for (String storageLocationName : storageLocationNames) { + PicklistOption picklistOption = new PicklistOption(); + picklistOption.label = storageLocationName; + picklistOption.value = storageLocationName; + + picklistOptions.add(picklistOption); + } + return picklistOptions; + } + + private static List getPurgeActionOptions() { + List logPurgeActionNames = new List{ DELETE_LOG_PURGE_ACTION }; + for (LoggerParameter__mdt matchingAdditionalParameter : LoggerParameter.matchOnPrefix(CUSTOM_LOG_PURGE_ACTION_PREFIX)) { + logPurgeActionNames.add(matchingAdditionalParameter.Value__c); + } + logPurgeActionNames.sort(); + List picklistOptions = initializePicklistOptions(); + for (String logPurgeActionName : logPurgeActionNames) { + PicklistOption picklistOption = new PicklistOption(); + picklistOption.label = logPurgeActionName; + picklistOption.value = logPurgeActionName; + + picklistOptions.add(picklistOption); + } + return picklistOptions; + } + + private static List getSaveMethodOptions() { + List saveMethodNames = new List(); for (Logger.SaveMethod saveMethod : Logger.SaveMethod.values()) { + saveMethodNames.add(saveMethod.name()); + } + for (LoggerParameter__mdt matchingAdditionalParameter : LoggerParameter.matchOnPrefix(CUSTOM_SAVE_METHOD_PREFIX)) { + saveMethodNames.add(matchingAdditionalParameter.Value__c); + } + saveMethodNames.sort(); + + List picklistOptions = initializePicklistOptions(); + for (String saveMethodName : saveMethodNames) { PicklistOption picklistOption = new PicklistOption(); - picklistOption.label = saveMethod.name(); - picklistOption.value = saveMethod.name(); + picklistOption.label = saveMethodName; + picklistOption.value = saveMethodName; picklistOptions.add(picklistOption); } @@ -283,6 +341,10 @@ public without sharing class LoggerSettingsController { @AuraEnabled public List loggingLevelOptions; @AuraEnabled + public List platformEventStorageLocationOptions; + @AuraEnabled + public List purgeActionOptions; + @AuraEnabled public List saveMethodOptions; @AuraEnabled public List setupOwnerTypeOptions; diff --git a/nebula-logger/core/main/configuration/customPermissions/CanModifyLoggerSettings.customPermission-meta.xml b/nebula-logger/core/main/log-management/customPermissions/CanModifyLoggerSettings.customPermission-meta.xml similarity index 100% rename from nebula-logger/core/main/configuration/customPermissions/CanModifyLoggerSettings.customPermission-meta.xml rename to nebula-logger/core/main/log-management/customPermissions/CanModifyLoggerSettings.customPermission-meta.xml diff --git a/nebula-logger/core/main/log-management/flexipages/LogEntryRecordPage.flexipage-meta.xml b/nebula-logger/core/main/log-management/flexipages/LogEntryRecordPage.flexipage-meta.xml index 9b65875f7..fec8cef86 100644 --- a/nebula-logger/core/main/log-management/flexipages/LogEntryRecordPage.flexipage-meta.xml +++ b/nebula-logger/core/main/log-management/flexipages/LogEntryRecordPage.flexipage-meta.xml @@ -293,6 +293,16 @@ RecordDatabaseResultType__cField + + + + uiBehavior + readonly + + Record.DatabaseResultCollectionSize__c + RecordDatabaseResultCollectionSize_cField1 + + @@ -341,6 +351,16 @@ RecordRecordSObjectType__cField + + + + uiBehavior + readonly + + Record.RecordCollectionSize__c + RecordRecordCollectionSize_cField1 + + @@ -439,6 +459,142 @@ Facet-f6d04c17-cbcd-40c6-aa58-c46313ecd8c4 Facet + + + + + uiBehavior + readonly + + Record.HttpRequestEndpoint__c + RecordHttpRequestEndpoint_cField1 + + + + + + uiBehavior + readonly + + Record.HttpRequestMethod__c + RecordHttpRequestMethod_cField1 + + + + + + uiBehavior + readonly + + Record.HttpRequestCompressed__c + RecordHttpRequestCompressed_cField1 + + + + + + uiBehavior + readonly + + Record.HttpRequestBody__c + RecordHttpRequestBody_cField1 + + + + + + uiBehavior + readonly + + Record.HttpRequestBodyMasked__c + RecordHttpRequestBodyMasked_cField1 + + + Facet-4cea44fd-c4f1-41ed-bada-eace6229f158 + Facet + + + + + + body + Facet-4cea44fd-c4f1-41ed-bada-eace6229f158 + + flexipage:column + flexipage_column1 + + + Facet-25eaa065-25af-4efc-9ce5-9d289a0fc3f0 + Facet + + + + + + uiBehavior + readonly + + Record.HttpResponseStatusCode__c + RecordHttpResponseStatusCode_cField1 + + + + + + uiBehavior + readonly + + Record.HttpResponseStatus__c + RecordHttpResponseStatus_cField1 + + + + + + uiBehavior + readonly + + Record.HttpResponseHeaderKeys__c + RecordHttpResponseHeaderKeys_cField1 + + + + + + uiBehavior + readonly + + Record.HttpResponseBody__c + RecordHttpResponseBody_cField1 + + + + + + uiBehavior + readonly + + Record.HttpResponseBodyMasked__c + RecordHttpResponseBodyMasked_cField1 + + + Facet-1f2467a3-2a30-4aad-ace3-80afc0a517d2 + Facet + + + + + + body + Facet-1f2467a3-2a30-4aad-ace3-80afc0a517d2 + + flexipage:column + flexipage_column15 + + + Facet-6395ceae-5694-468a-911f-714456910736 + Facet + @@ -537,6 +693,46 @@ + + + + columns + Facet-25eaa065-25af-4efc-9ce5-9d289a0fc3f0 + + + label + HTTP Request + + flexipage:fieldSection + flexipage_fieldSection1 + + + {!Record.HttpRequestEndpoint__c} + NE + + + + + + + + columns + Facet-6395ceae-5694-468a-911f-714456910736 + + + label + HTTP Response + + flexipage:fieldSection + flexipage_fieldSection10 + + + {!Record.HttpResponseStatus__c} + NE + + + + Facet-7beef2bc-28d3-490f-96b8-7a8f85424802 Facet diff --git a/nebula-logger/core/main/log-management/flexipages/LogRecordPage.flexipage-meta.xml b/nebula-logger/core/main/log-management/flexipages/LogRecordPage.flexipage-meta.xml index 974483a16..f5c97a038 100644 --- a/nebula-logger/core/main/log-management/flexipages/LogRecordPage.flexipage-meta.xml +++ b/nebula-logger/core/main/log-management/flexipages/LogRecordPage.flexipage-meta.xml @@ -324,6 +324,16 @@ RecordLogRetentionDate__cField + + + + uiBehavior + readonly + + Record.LogPurgeAction__c + RecordLogPurgeAction_cField1 + + Facet-0d174e01-17f8-4c93-8f5a-8983346600f8 Facet diff --git a/nebula-logger/core/main/log-management/layouts/LogEntry__c-Log Entry Layout.layout-meta.xml b/nebula-logger/core/main/log-management/layouts/LogEntry__c-Log Entry Layout.layout-meta.xml index 80497e3e1..af7eae2f0 100644 --- a/nebula-logger/core/main/log-management/layouts/LogEntry__c-Log Entry Layout.layout-meta.xml +++ b/nebula-logger/core/main/log-management/layouts/LogEntry__c-Log Entry Layout.layout-meta.xml @@ -235,6 +235,10 @@ Readonly DatabaseResultType__c + + Edit + DatabaseResultCollectionSize__c + Readonly DatabaseResultJson__c @@ -256,6 +260,10 @@ Readonly RecordSObjectType__c + + Readonly + RecordCollectionSize__c + Readonly RecordId__c @@ -283,6 +291,56 @@ + + true + true + true + + + + Readonly + HttpRequestEndpoint__c + + + Readonly + HttpRequestMethod__c + + + Readonly + HttpRequestCompressed__c + + + Readonly + HttpRequestBody__c + + + + + + true + true + true + + + + Readonly + HttpResponseStatusCode__c + + + Readonly + HttpResponseStatus__c + + + Readonly + HttpResponseHeaderKeys__c + + + Readonly + HttpResponseBody__c + + + + true true @@ -397,7 +455,7 @@ false false - 00h1h000002oBf9 + 00h1k000003yap8 4 0 Default diff --git a/nebula-logger/core/main/log-management/layouts/Log__c-Log Layout.layout-meta.xml b/nebula-logger/core/main/log-management/layouts/Log__c-Log Layout.layout-meta.xml index 161423505..47c5814bb 100644 --- a/nebula-logger/core/main/log-management/layouts/Log__c-Log Layout.layout-meta.xml +++ b/nebula-logger/core/main/log-management/layouts/Log__c-Log Layout.layout-meta.xml @@ -100,6 +100,10 @@ Readonly LogRetentionDate__c + + Readonly + LogPurgeAction__c + @@ -385,7 +389,7 @@ false false - 00h1F000005MrCM + 00h1F000006uWJf 4 0 Default diff --git a/nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/data/getLogEntryEventSchema.json b/nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/data/getSchemaForName.json similarity index 100% rename from nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/data/getLogEntryEventSchema.json rename to nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/data/getSchemaForName.json diff --git a/nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/logEntryEventStream.test.js b/nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/logEntryEventStream.test.js index 1dff298ab..7c4369e7d 100644 --- a/nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/logEntryEventStream.test.js +++ b/nebula-logger/core/main/log-management/lwc/logEntryEventStream/__tests__/logEntryEventStream.test.js @@ -1,7 +1,7 @@ import { createElement } from 'lwc'; import LogEntryEventStream from 'c/logEntryEventStream'; import { jestMockPublish } from 'lightning/empApi'; -import getLogEntryEventSchema from '@salesforce/apex/LoggerSObjectMetadata.getLogEntryEventSchema'; +import getSchemaForName from '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName'; const loggingLevels = { FINEST: 2, @@ -25,10 +25,10 @@ const mockLogEntryEventTemplate = { TransactionEntryNumber__c: 1 }; -const mockLogEntryEventSchemaTemplate = require('./data/getLogEntryEventSchema.json'); +const mockLogEntryEventSchemaTemplate = require('./data/getSchemaForName.json'); jest.mock( - '@salesforce/apex/LoggerSObjectMetadata.getLogEntryEventSchema', + '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName', () => { return { default: jest.fn() @@ -39,7 +39,7 @@ jest.mock( async function createStreamElement(namespace) { const mockLogEntryEventSchema = generateLogEntryEventSchema(namespace); - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchema); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchema); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream }); @@ -210,7 +210,7 @@ describe('LogEntryEventStream tests', () => { ); }); it('includes matching log entry event for origin type filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -235,7 +235,7 @@ describe('LogEntryEventStream tests', () => { expect(eventStreamDiv.textContent).toBe(expectedStreamText); }); it('excludes non-matching log entry event for origin type filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -259,7 +259,7 @@ describe('LogEntryEventStream tests', () => { expect(eventStreamDiv.textContent).toBeFalsy(); }); it('includes matching log entry event for origin location filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -284,7 +284,7 @@ describe('LogEntryEventStream tests', () => { expect(eventStreamDiv.textContent).toBe(expectedStreamText); }); it('excludes non-matching log entry event for origin location filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -308,7 +308,7 @@ describe('LogEntryEventStream tests', () => { expect(eventStreamDiv.textContent).toBeFalsy(); }); it('includes matching log entry event for logged by filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -333,7 +333,7 @@ describe('LogEntryEventStream tests', () => { expect(eventStreamDiv.textContent).toBe(expectedStreamText); }); it('excludes non-matching log entry event for logged by filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -357,7 +357,7 @@ describe('LogEntryEventStream tests', () => { expect(eventStreamDiv.textContent).toBeFalsy(); }); it('includes matching log entry event using string for message filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -382,7 +382,7 @@ describe('LogEntryEventStream tests', () => { expect(eventStreamDiv.textContent).toBe(expectedStreamText); }); it('excludes non-matching log entry event for message filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream @@ -407,7 +407,7 @@ describe('LogEntryEventStream tests', () => { }); it('includes matching log entry event using regex for message filter', async () => { - getLogEntryEventSchema.mockResolvedValue(mockLogEntryEventSchemaTemplate); + getSchemaForName.mockResolvedValue(mockLogEntryEventSchemaTemplate); const element = createElement('log-entry-event-stream', { is: LogEntryEventStream diff --git a/nebula-logger/core/main/log-management/lwc/logEntryEventStream/logEntryEventStream.js b/nebula-logger/core/main/log-management/lwc/logEntryEventStream/logEntryEventStream.js index 5ee6adb6e..272b3aa56 100644 --- a/nebula-logger/core/main/log-management/lwc/logEntryEventStream/logEntryEventStream.js +++ b/nebula-logger/core/main/log-management/lwc/logEntryEventStream/logEntryEventStream.js @@ -5,7 +5,7 @@ import { LightningElement } from 'lwc'; import { subscribe, unsubscribe } from 'lightning/empApi'; -import getLogEntryEventSchema from '@salesforce/apex/LoggerSObjectMetadata.getLogEntryEventSchema'; +import getSchemaForName from '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName'; export default class LogEntryEventStream extends LightningElement { unfilteredEvents = []; @@ -29,7 +29,7 @@ export default class LogEntryEventStream extends LightningElement { async connectedCallback() { document.title = 'Log Entry Event Stream'; - getLogEntryEventSchema().then(result => { + getSchemaForName({ sobjectApiName: 'LogEntryEvent__e' }).then(result => { this._logEntryEventSchema = result; this._channel = '/event/' + this._logEntryEventSchema.apiName; diff --git a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getObjectInfo.json b/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getObjectInfo.json deleted file mode 100644 index 864becc9d..000000000 --- a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getObjectInfo.json +++ /dev/null @@ -1,822 +0,0 @@ -{ - "apiName": "LoggerSettings__c", - "associateEntityType": null, - "associateParentEntity": null, - "childRelationships": [ - { - "childObjectApiName": "AttachedContentDocument", - "fieldName": "LinkedEntityId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "AttachedContentDocuments" - }, - { - "childObjectApiName": "CollaborationGroupRecord", - "fieldName": "RecordId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "RecordAssociatedGroups" - }, - { - "childObjectApiName": "CombinedAttachment", - "fieldName": "ParentId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "CombinedAttachments" - }, - { - "childObjectApiName": "ContactRequest", - "fieldName": "WhatId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "ContactRequests" - }, - { - "childObjectApiName": "ContentDocumentLink", - "fieldName": "LinkedEntityId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "ContentDocumentLinks" - }, - { - "childObjectApiName": "DuplicateRecordItem", - "fieldName": "RecordId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "DuplicateRecordItems" - }, - { - "childObjectApiName": "EntitySubscription", - "fieldName": "ParentId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "FeedSubscriptionsForEntity" - }, - { - "childObjectApiName": "NetworkActivityAudit", - "fieldName": "ParentEntityId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "ParentEntities" - }, - { - "childObjectApiName": "NetworkUserHistoryRecent", - "fieldName": "RecordId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "NetworkUserHistoryRecentToRecord" - }, - { - "childObjectApiName": "ProcessException", - "fieldName": "AttachedToId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "ProcessExceptions" - }, - { - "childObjectApiName": "ProcessInstance", - "fieldName": "TargetObjectId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "ProcessInstances" - }, - { - "childObjectApiName": "ProcessInstanceHistory", - "fieldName": "TargetObjectId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "ProcessSteps" - }, - { - "childObjectApiName": "RecordAction", - "fieldName": "RecordId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "RecordActions" - }, - { - "childObjectApiName": "RecordActionHistory", - "fieldName": "ParentRecordId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "RecordActionHistories" - }, - { - "childObjectApiName": "TopicAssignment", - "fieldName": "EntityId", - "junctionIdListNames": [], - "junctionReferenceTo": [], - "relationshipName": "TopicAssignments" - } - ], - "createable": true, - "custom": true, - "defaultRecordTypeId": "012000000000000AAA", - "deletable": true, - "dependentFields": {}, - "feedEnabled": false, - "fields": { - "CloneSourceId": { - "apiName": "CloneSourceId", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "Reference", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Clone Source", - "length": 18, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": true, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": false, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - }, - "CreatedById": { - "apiName": "CreatedById", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "Reference", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Created By ID", - "length": 18, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": true, - "referenceTargetField": null, - "referenceToInfos": [ - { - "apiName": "User", - "nameFields": ["FirstName", "LastName", "Name"] - } - ], - "relationshipName": "CreatedBy", - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - }, - "CreatedDate": { - "apiName": "CreatedDate", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "DateTime", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Created Date", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - }, - "DefaultLogShareAccessLevel__c": { - "apiName": "DefaultLogShareAccessLevel__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "String", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "Uses Apex managed sharing to grants users read or edit access to their log records (on insert only). When no access level is specified, no Apex sharing logic is executed. This only gives record-level access - users will still need to be granted access to the Log__c object using permission sets or profiles.\n\nPossible Values:\n(blank)\nRead\nEdit", - "label": "Log Access Level", - "length": 255, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": false, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "DefaultNumberOfDaysToRetainLogs__c": { - "apiName": "DefaultNumberOfDaysToRetainLogs__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "Double", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "This value is used to set the field Log__c.LogRetentionDate__c, which is then used by LogBatchPurger to delete old logs. To keep logs indefinitely, set this field to blank (null).", - "label": "Days to Retain Logs", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 4, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": false, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "DefaultSaveMethod__c": { - "apiName": "DefaultSaveMethod__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "String", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "Defaults to 'EVENT_BUS'. This controls the default save method used by Logger when calling saveLog().\n\nPossible values:\nEVENT_BUS\nQUEUEABLE\nREST\nSYNCHRONOUS_DML", - "label": "Save Method", - "length": 255, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "Id": { - "apiName": "Id", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "String", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Record ID", - "length": 18, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - }, - "IsAnonymousModeEnabled__c": { - "apiName": "IsAnonymousModeEnabled__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "Boolean", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "When enabled, any logs generated will not have any user-specific details set - any fields related to the User, Profile, etc. will be null.\n\nNote: this feature only works properly when using the save method EVENT_BUS.", - "label": "Enable Anonymous Mode", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "IsApexSystemDebugLoggingEnabled__c": { - "apiName": "IsApexSystemDebugLoggingEnabled__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "Boolean", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "When enabled, Logger will automatically call Apex's System.debug(). To help with performance, this option should be disabled in production unless you are actively troubleshooting an issue.", - "label": "Enable Apex System.debug()", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "IsJavaScriptConsoleLoggingEnabled__c": { - "apiName": "IsJavaScriptConsoleLoggingEnabled__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "Boolean", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "When enabled, Logger will automatically call the browser's console.log() function when logging via lightning components. To help with performance, this option should be disabled in production unless you are actively troubleshooting an issue.", - "label": "Enable JavaScript console.log()", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "IsDataMaskingEnabled__c": { - "apiName": "IsDataMaskingEnabled__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "Boolean", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "When enabled, any data-mask rules (configured in LogEntryDataMaskRule__mdt) will be automatically applied to log entry messages.", - "label": "Enable Data Masking", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "IsDeleted": { - "apiName": "IsDeleted", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "Boolean", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Deleted", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - }, - "IsEnabled__c": { - "apiName": "IsEnabled__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "Boolean", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "Controls if Logger is enabled for the specified level (organization, profile, or user)", - "label": "Enabled", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "LastModifiedById": { - "apiName": "LastModifiedById", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "Reference", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Last Modified By ID", - "length": 18, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": true, - "referenceTargetField": null, - "referenceToInfos": [ - { - "apiName": "User", - "nameFields": ["FirstName", "LastName", "Name"] - } - ], - "relationshipName": "LastModifiedBy", - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - }, - "LastModifiedDate": { - "apiName": "LastModifiedDate", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "DateTime", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Last Modified Date", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - }, - "LoggingLevel__c": { - "apiName": "LoggingLevel__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "String", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "Possible values:\nERROR\nWARN\nINFO\nDEBUG\nFINE\nFINER\nFINEST", - "label": "Logging Level", - "length": 255, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "Name": { - "apiName": "Name", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": false, - "dataType": "String", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Name", - "length": 80, - "nameField": true, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": false, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "SetupOwnerId": { - "apiName": "SetupOwnerId", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": false, - "dataType": "Reference", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "Location", - "length": 18, - "nameField": false, - "polymorphicForeignKey": true, - "precision": 0, - "reference": true, - "referenceTargetField": null, - "referenceToInfos": [ - { - "apiName": "Organization", - "nameFields": ["Name"] - }, - { - "apiName": "Profile", - "nameFields": ["Name"] - }, - { - "apiName": "User", - "nameFields": ["FirstName", "LastName", "Name"] - } - ], - "relationshipName": "SetupOwner", - "required": false, - "scale": 0, - "searchPrefilterable": true, - "sortable": true, - "unique": false, - "updateable": true - }, - "StripInaccessibleRecordFields__c": { - "apiName": "StripInaccessibleRecordFields__c", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": true, - "custom": true, - "dataType": "Boolean", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": "When enabled, any time an SObject record is logged, only fields that the current user can access will be included in the record's JSON.", - "label": "Strip Inaccessible Record Fields", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": true - }, - "SystemModstamp": { - "apiName": "SystemModstamp", - "calculated": false, - "compound": false, - "compoundComponentName": null, - "compoundFieldName": null, - "controllerName": null, - "controllingFields": [], - "createable": false, - "custom": false, - "dataType": "DateTime", - "extraTypeInfo": null, - "filterable": true, - "filteredLookupInfo": null, - "highScaleNumber": false, - "htmlFormatted": false, - "inlineHelpText": null, - "label": "System Modstamp", - "length": 0, - "nameField": false, - "polymorphicForeignKey": false, - "precision": 0, - "reference": false, - "referenceTargetField": null, - "referenceToInfos": [], - "relationshipName": null, - "required": true, - "scale": 0, - "searchPrefilterable": false, - "sortable": true, - "unique": false, - "updateable": false - } - }, - "keyPrefix": "a03", - "label": "Logger Settings", - "labelPlural": "Logger Settings", - "layoutable": true, - "mruEnabled": false, - "nameFields": ["Name"], - "queryable": true, - "recordTypeInfos": { - "012000000000000AAA": { - "available": true, - "defaultRecordTypeMapping": true, - "master": true, - "name": "Master", - "recordTypeId": "012000000000000AAA" - } - }, - "searchable": true, - "themeInfo": null, - "updateable": true -} diff --git a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getPicklistOptions.json b/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getPicklistOptions.json index c27a0c97d..3c2608b42 100644 --- a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getPicklistOptions.json +++ b/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getPicklistOptions.json @@ -9,6 +9,14 @@ { "value": "FINER", "label": "FINER" }, { "value": "FINEST", "label": "FINEST" } ], + "platformEventStorageLocationOptions": [ + { "value": "", "label": "--None--" }, + { "value": "CUSTOM_OBJECTS", "label": "CUSTOM_OBJECTS" } + ], + "purgeActionOptions": [ + { "value": "", "label": "--None--" }, + { "value": "Delete", "label": "Delete" } + ], "saveMethodOptions": [ { "value": "", "label": "--None--" }, { "value": "EVENT_BUS", "label": "EVENT_BUS" }, diff --git a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getLoggerSettingsSchema.json b/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getSchemaForName.json similarity index 93% rename from nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getLoggerSettingsSchema.json rename to nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getSchemaForName.json index b6512d296..ef46c0729 100644 --- a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getLoggerSettingsSchema.json +++ b/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/data/getSchemaForName.json @@ -76,6 +76,13 @@ "localApiName": "DefaultNumberOfDaysToRetainLogs__c", "type": "Double" }, + "DefaultPlatformEventStorageLocation__c": { + "apiName": "DefaultPlatformEventStorageLocation__c", + "inlineHelpText": "Defaults to CUSTOM_OBJECTS. This controls the default location where LogEntryEvent__e records are stored - when null, LogEntryEvent__e records will not be stored.", + "label": "Platform Event Storage Location", + "localApiName": "DefaultPlatformEventStorageLocation__c", + "type": "string" + }, "DefaultSaveMethod__c": { "apiName": "DefaultSaveMethod__c", "inlineHelpText": "Defaults to EVENT_BUS. This controls the default save method used by Logger when calling saveLog(). In most situations, EVENT_BUS should be used.", @@ -118,13 +125,6 @@ "localApiName": "IsJavaScriptConsoleLoggingEnabled__c", "type": "Boolean" }, - "IsPlatformEventStorageEnabled__c": { - "apiName": "IsPlatformEventStorageEnabled__c", - "inlineHelpText": "Controls if LogEntryEvent__e platform events are transformed & stored in the custom objects Log__c and LogEntry__c (when IsSavingEnabled__c == true).", - "label": "Store Platform Events", - "localApiName": "IsPlatformEventStorageEnabled__c", - "type": "Boolean" - }, "IsRecordFieldStrippingEnabled__c": { "apiName": "IsRecordFieldStrippingEnabled__c", "inlineHelpText": "When enabled, any time an SObject record is logged, only fields that the current user can access will be included in the record's JSON.", diff --git a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/loggerSettings.test.js b/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/loggerSettings.test.js index ddd4453db..30d0df456 100644 --- a/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/loggerSettings.test.js +++ b/nebula-logger/core/main/log-management/lwc/loggerSettings/__tests__/loggerSettings.test.js @@ -4,7 +4,7 @@ import LoggerSettings from 'c/loggerSettings'; // LoggerSettings__c metadata import canUserModifyLoggerSettings from '@salesforce/apex/LoggerSettingsController.canUserModifyLoggerSettings'; -import getLoggerSettingsSchema from '@salesforce/apex/LoggerSObjectMetadata.getLoggerSettingsSchema'; +import getSchemaForName from '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName'; import getPicklistOptions from '@salesforce/apex/LoggerSettingsController.getPicklistOptions'; import getOrganization from '@salesforce/apex/LoggerSettingsController.getOrganization'; import searchForSetupOwner from '@salesforce/apex/LoggerSettingsController.searchForSetupOwner'; @@ -16,7 +16,7 @@ import saveRecord from '@salesforce/apex/LoggerSettingsController.saveRecord'; import deleteRecord from '@salesforce/apex/LoggerSettingsController.deleteRecord'; // Mock metadata -const mockLoggerSettingsSchema = require('./data/getLoggerSettingsSchema.json'); +const mockLoggerSettingsSchema = require('./data/getSchemaForName.json'); const mockOrganization = require('./data/getOrganization.json'); const mockPicklistOptions = require('./data/getPicklistOptions.json'); @@ -106,7 +106,7 @@ jest.mock( ); jest.mock( - '@salesforce/apex/LoggerSObjectMetadata.getLoggerSettingsSchema', + '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName', () => { return { default: jest.fn() @@ -118,7 +118,7 @@ jest.mock( async function initializeElement(enableModifyAccess) { // Assign mock values for resolved Apex promises canUserModifyLoggerSettings.mockResolvedValue(enableModifyAccess); - getLoggerSettingsSchema.mockResolvedValue(mockLoggerSettingsSchema); + getSchemaForName.mockResolvedValue(mockLoggerSettingsSchema); getPicklistOptions.mockResolvedValue(mockPicklistOptions); getRecords.mockResolvedValue(mockRecords); @@ -165,7 +165,7 @@ describe('Logger Settings lwc tests', () => { // Verify the expected Apex/framework calls expect(canUserModifyLoggerSettings).toHaveBeenCalledTimes(1); - expect(getLoggerSettingsSchema).toHaveBeenCalledTimes(1); + expect(getSchemaForName).toHaveBeenCalledTimes(1); expect(getPicklistOptions).toHaveBeenCalledTimes(1); expect(getRecords).toHaveBeenCalledTimes(1); expect(createRecord).toHaveBeenCalledTimes(0); diff --git a/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettings.js b/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettings.js index 1d6a48835..0763535a7 100644 --- a/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettings.js +++ b/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettings.js @@ -10,7 +10,7 @@ import { ShowToastEvent } from 'lightning/platformShowToastEvent'; // LoggerSettings__c metadata import { generatePageLayout } from './loggerSettingsPageLayout'; import canUserModifyLoggerSettings from '@salesforce/apex/LoggerSettingsController.canUserModifyLoggerSettings'; -import getLoggerSettingsSchema from '@salesforce/apex/LoggerSObjectMetadata.getLoggerSettingsSchema'; +import getSchemaForName from '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName'; import getPicklistOptions from '@salesforce/apex/LoggerSettingsController.getPicklistOptions'; import getOrganization from '@salesforce/apex/LoggerSettingsController.getOrganization'; import searchForSetupOwner from '@salesforce/apex/LoggerSettingsController.searchForSetupOwner'; @@ -50,7 +50,7 @@ export default class LoggerSettings extends LightningElement { connectedCallback() { document.title = this.title; this.showLoadingSpinner = true; - Promise.all([getOrganization(), getLoggerSettingsSchema(), getPicklistOptions(), canUserModifyLoggerSettings()]) + Promise.all([getOrganization(), getSchemaForName({ sobjectApiName: 'LoggerSettings__c' }), getPicklistOptions(), canUserModifyLoggerSettings()]) .then(([organizationRecordResult, loggerSettingsSchemaResult, apexPicklistOptionsResult, canUserModifyLoggerSettingsResult]) => { this.organization = organizationRecordResult; this._loggerSettingsSchema = loggerSettingsSchemaResult; @@ -118,22 +118,6 @@ export default class LoggerSettings extends LightningElement { ? event.target.checked : event.target.value; this._currentRecord[fieldApiName] = fieldValue; - if (fieldValue && fieldApiName === this._loggerSettingsSchema.fields.IsSavingEnabled__c.localApiName) { - const storageEnabledCheckbox = this.template.querySelector( - `[data-id="${this._loggerSettingsSchema.fields.IsPlatformEventStorageEnabled__c.localApiName}"]` - ); - if (storageEnabledCheckbox) { - storageEnabledCheckbox.checked = true; - } - this.handleFieldChange({ - target: { - type: 'checkbox', - checked: true, - dataset: { id: this._loggerSettingsSchema.fields.IsPlatformEventStorageEnabled__c.localApiName } - } - }); - } - this._setIsNewOrganizationRecord(); this._setShowSetupOwnerLookup(); } @@ -335,7 +319,7 @@ export default class LoggerSettings extends LightningElement { 'LoggingLevel__c', 'IsSavingEnabled__c', 'DefaultSaveMethod__c', - 'IsPlatformEventStorageEnabled__c', + 'DefaultPlatformEventStorageLocation__c', 'DefaultNumberOfDaysToRetainLogs__c', 'DefaultLogOwner__c' ]; diff --git a/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettingsPageLayout.js b/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettingsPageLayout.js index c88baf4da..2675c3e2d 100644 --- a/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettingsPageLayout.js +++ b/nebula-logger/core/main/log-management/lwc/loggerSettings/loggerSettingsPageLayout.js @@ -13,7 +13,7 @@ const PAGE_LAYOUT_CONFIG = { showInEditMode: true, columns: [ { - fieldApiNames: ['IsSavingEnabled__c', 'DefaultSaveMethod__c', 'IsPlatformEventStorageEnabled__c'], + fieldApiNames: ['IsSavingEnabled__c', 'DefaultSaveMethod__c', 'DefaultPlatformEventStorageLocation__c'], size: 6 }, { @@ -41,8 +41,8 @@ const PAGE_LAYOUT_CONFIG = { showInReadOnlyMode: true, showInEditMode: true, columns: [ - { fieldApiNames: ['DefaultNumberOfDaysToRetainLogs__c', 'DefaultLogScenario__c'], size: 6 }, - { fieldApiNames: ['DefaultLogShareAccessLevel__c', 'DefaultLogOwner__c'], size: 6 } + { fieldApiNames: ['DefaultNumberOfDaysToRetainLogs__c', 'DefaultLogPurgeAction__c'], size: 6 }, + { fieldApiNames: ['DefaultLogScenario__c', 'DefaultLogShareAccessLevel__c', 'DefaultLogOwner__c'], size: 6 } ] } ] @@ -59,6 +59,8 @@ const LoggerSettingsPageLayout = class { // TODO - long term, this feels like the wrong place for this mapping to live, but it'll live here for now const picklistOptions = { LoggingLevel__c: apexPicklistOptions.loggingLevelOptions, + DefaultLogPurgeAction__c: apexPicklistOptions.purgeActionOptions, + DefaultPlatformEventStorageLocation__c: apexPicklistOptions.platformEventStorageLocationOptions, DefaultSaveMethod__c: apexPicklistOptions.saveMethodOptions, DefaultLogShareAccessLevel__c: apexPicklistOptions.shareAccessLevelOptions }; diff --git a/nebula-logger/core/main/log-management/lwc/relatedLogEntries/__tests__/relatedLogEntries.test.js b/nebula-logger/core/main/log-management/lwc/relatedLogEntries/__tests__/relatedLogEntries.test.js index be3658acd..bdad494e7 100644 --- a/nebula-logger/core/main/log-management/lwc/relatedLogEntries/__tests__/relatedLogEntries.test.js +++ b/nebula-logger/core/main/log-management/lwc/relatedLogEntries/__tests__/relatedLogEntries.test.js @@ -29,6 +29,7 @@ describe('Related Log Entries lwc tests', () => { while (document.body.firstChild) { document.body.removeChild(document.body.firstChild); } + jest.clearAllMocks(); }); it('sets query result', async () => { diff --git a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryOrigin__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryOrigin__c.field-meta.xml index 78c2f66c6..98d394fce 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryOrigin__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryOrigin__c.field-meta.xml @@ -1,11 +1,13 @@ LogEntryOrigin__c + Active false LogEntry__r.Origin__c BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryTimestamp__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryTimestamp__c.field-meta.xml index 5d0cf619c..e39d27809 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryTimestamp__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntryTimestamp__c.field-meta.xml @@ -1,11 +1,13 @@ LogEntryTimestamp__c + Active false LogEntry__r.Timestamp__c BlankAsZero false + Confidential false false DateTime diff --git a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntry__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntry__c.field-meta.xml index 9647cd674..b0f1225e0 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntry__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogEntry__c.field-meta.xml @@ -1,6 +1,7 @@ LogEntry__c + Active false LogEntry__c @@ -8,6 +9,7 @@ LogEntryTags 0 false + Confidential true false false diff --git a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogLink__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogLink__c.field-meta.xml index 1a8e66b7d..9459bcfe4 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogLink__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LogLink__c.field-meta.xml @@ -1,6 +1,7 @@ LogLink__c + Active false HYPERLINK( LogEntry__r.Log__c, @@ -10,6 +11,7 @@ BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LoggedByUsernameLink__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LoggedByUsernameLink__c.field-meta.xml index ca4a91e7a..5ccff53fd 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LoggedByUsernameLink__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/LoggedByUsernameLink__c.field-meta.xml @@ -1,11 +1,14 @@ LoggedByUsernameLink__c + Active + PII false LogEntry__r.LoggedByUsernameLink__c BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/Tag__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/Tag__c.field-meta.xml index e497a3092..7cc71acff 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/Tag__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/Tag__c.field-meta.xml @@ -1,6 +1,7 @@ Tag__c + Active false LoggerTag__c @@ -8,6 +9,7 @@ LogEntryTags 1 false + Confidential true false false diff --git a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/UniqueId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/UniqueId__c.field-meta.xml index d1d0d468b..2740832f8 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/UniqueId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntryTag__c/fields/UniqueId__c.field-meta.xml @@ -1,6 +1,7 @@ UniqueId__c + Active false An external ID field used to ensure that Log Entry Tag records are unique true @@ -8,6 +9,7 @@ 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/DatabaseResultCollectionSize__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/DatabaseResultCollectionSize__c.field-meta.xml new file mode 100644 index 000000000..a7e1b78cf --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/DatabaseResultCollectionSize__c.field-meta.xml @@ -0,0 +1,15 @@ + + + DatabaseResultCollectionSize__c + The number of items contained in the collection of database results + false + The number of items contained in the collection of database results + + 10 + false + 0 + false + false + Number + false + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/EventUuid__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/EventUuid__c.field-meta.xml index 141e35b30..66f4977f2 100644 --- a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/EventUuid__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/EventUuid__c.field-meta.xml @@ -11,7 +11,7 @@ For more details, refer to https://developer.salesforce.com/docs/atlas.en-us.pla For more details, refer to https://developer.salesforce.com/docs/atlas.en-us.platform_events.meta/platform_events/platform_events_event_uuid.htm 36 - false + false false false Text diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBodyMasked__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBodyMasked__c.field-meta.xml new file mode 100644 index 000000000..462c7d9e3 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBodyMasked__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpRequestBodyMasked__c + false + false + Indicates if sensitive data was removed from the HTTP Request Body, using log entry data mask rules + + false + false + Checkbox + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBody__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBody__c.field-meta.xml new file mode 100644 index 000000000..50414a97a --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestBody__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpRequestBody__c + false + + 131072 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestCompressed__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestCompressed__c.field-meta.xml new file mode 100644 index 000000000..ebf8828f6 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestCompressed__c.field-meta.xml @@ -0,0 +1,10 @@ + + + HttpRequestCompressed__c + false + false + + false + false + Checkbox + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestEndpoint__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestEndpoint__c.field-meta.xml new file mode 100644 index 000000000..f50bbbd77 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestEndpoint__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpRequestEndpoint__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestMethod__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestMethod__c.field-meta.xml new file mode 100644 index 000000000..6bfb10c20 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpRequestMethod__c.field-meta.xml @@ -0,0 +1,51 @@ + + + HttpRequestMethod__c + false + + false + false + false + Picklist + + false + + false + + DELETE + false + + + + GET + false + + + + HEAD + false + + + + PATCH + false + + + + POST + false + + + + PUT + false + + + + TRACE + false + + + + + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBodyMasked__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBodyMasked__c.field-meta.xml new file mode 100644 index 000000000..012766818 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBodyMasked__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseBodyMasked__c + false + false + Indicates if sensitive data was removed from the HTTP Response Body, using log entry data mask rules + + false + false + Checkbox + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBody__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBody__c.field-meta.xml new file mode 100644 index 000000000..b0f892ac1 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseBody__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseBody__c + false + + 131072 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseHeaderKeys__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseHeaderKeys__c.field-meta.xml new file mode 100644 index 000000000..d5ca1263f --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseHeaderKeys__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseHeaderKeys__c + false + + 1000 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatusCode__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatusCode__c.field-meta.xml new file mode 100644 index 000000000..d3f8431d7 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatusCode__c.field-meta.xml @@ -0,0 +1,13 @@ + + + HttpResponseStatusCode__c + false + + 10 + false + 0 + false + false + Number + false + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatus__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatus__c.field-meta.xml new file mode 100644 index 000000000..a12ff395e --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/HttpResponseStatus__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpResponseStatus__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/RecordCollectionSize__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/RecordCollectionSize__c.field-meta.xml new file mode 100644 index 000000000..62cfee8f6 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/LogEntry__c/fields/RecordCollectionSize__c.field-meta.xml @@ -0,0 +1,15 @@ + + + RecordCollectionSize__c + The number of items contained in the collection of records + false + The number of items contained in the collection of records + + 10 + false + 0 + false + false + Number + false + diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseNumber__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseNumber__c.field-meta.xml index 564a48ab6..8ecde6685 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseNumber__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseNumber__c.field-meta.xml @@ -1,12 +1,14 @@ ApiReleaseNumber__c + Active The release number for the org's instance - determined by making a callout to status.salesforce.com false The release number for the org's instance - determined by making a callout to status.salesforce.com 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseVersion__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseVersion__c.field-meta.xml index e4020ebec..aa6351894 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseVersion__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiReleaseVersion__c.field-meta.xml @@ -1,12 +1,14 @@ ApiReleaseVersion__c + Active The release version for the org's instance - determined by making a callout to status.salesforce.com false The release version for the org's instance - determined by making a callout to status.salesforce.com 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiVersion__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiVersion__c.field-meta.xml index ece640f1e..070d94666 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiVersion__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ApiVersion__c.field-meta.xml @@ -1,11 +1,13 @@ ApiVersion__c + Active The Salesforce release (API version) of the environment false The Salesforce release (API version) of the environment false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedBy__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedBy__c.field-meta.xml index f94b48d5e..ecaae413d 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedBy__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedBy__c.field-meta.xml @@ -1,12 +1,14 @@ ClosedBy__c + Active SetNull false User ClosedLogs false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedDate__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedDate__c.field-meta.xml index d9322c007..97479330f 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedDate__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ClosedDate__c.field-meta.xml @@ -1,9 +1,11 @@ ClosedDate__c + Active false false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/Comments__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/Comments__c.field-meta.xml index 332050596..78c68131a 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/Comments__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/Comments__c.field-meta.xml @@ -1,9 +1,11 @@ Comments__c + Active false 4000 + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/EndTime__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/EndTime__c.field-meta.xml index ad5568429..544ccaed1 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/EndTime__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/EndTime__c.field-meta.xml @@ -1,8 +1,10 @@ EndTime__c + Active false + Confidential LogEntry__c.Timestamp__c LogEntry__c.Log__c max diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/IsClosed__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/IsClosed__c.field-meta.xml index 1521dad6f..6ef326596 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/IsClosed__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/IsClosed__c.field-meta.xml @@ -1,9 +1,11 @@ IsClosed__c + Active false false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/IsResolved__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/IsResolved__c.field-meta.xml index 2fe9207bd..d80fd86bf 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/IsResolved__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/IsResolved__c.field-meta.xml @@ -1,9 +1,11 @@ IsResolved__c + Active false false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/Issue__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/Issue__c.field-meta.xml index 566c088a7..7175495e0 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/Issue__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/Issue__c.field-meta.xml @@ -1,9 +1,11 @@ Issue__c + Active false false + Confidential true true false @@ -11,7 +13,7 @@ true - true + false Code Issue false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/Locale__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/Locale__c.field-meta.xml index bb71557e3..d53d290f0 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/Locale__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/Locale__c.field-meta.xml @@ -1,10 +1,12 @@ Locale__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LogEntriesSummary__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogEntriesSummary__c.field-meta.xml index 946ebfbc1..50b0e7bde 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LogEntriesSummary__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogEntriesSummary__c.field-meta.xml @@ -1,6 +1,7 @@ LogEntriesSummary__c + Active false TEXT(TotalLogEntries__c) + ' total entries' + IF( @@ -16,6 +17,7 @@ BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LogPurgeAction__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogPurgeAction__c.field-meta.xml new file mode 100644 index 000000000..2bdef9af0 --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogPurgeAction__c.field-meta.xml @@ -0,0 +1,25 @@ + + + LogPurgeAction__c + Active + false + Used by the batch job LogBatchPurger, this indicates the action that should be taken for a Log__c record when LogRetentionDate__c < TODAY. Out of the box, only 'Delete' is used by Logger, but plugins can add & use new values. + + false + Confidential + true + true + false + Picklist + + + false + + Delete + false + + + + + diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LogRetentionDate__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogRetentionDate__c.field-meta.xml index 0b900d312..c29cf0550 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LogRetentionDate__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogRetentionDate__c.field-meta.xml @@ -1,6 +1,7 @@ LogRetentionDate__c + Active The date that this log can be automatically deleted by the batch job LogBatchPurger. It defaults to 2 weeks after creation (configurable in Logger Settings), but the date can be set manually or via automation if certain logs need to be kept longer/indefinitely. @@ -8,6 +9,7 @@ It defaults to 2 weeks after creation (configurable in Logger Settings), but the The date that this log can be automatically deleted by the batch job LogBatchPurger. false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsernameLink__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsernameLink__c.field-meta.xml index c87b7ff13..6ca70ce3e 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsernameLink__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsernameLink__c.field-meta.xml @@ -1,6 +1,8 @@ LoggedByUsernameLink__c + Active + PII false IF( ISBLANK(LoggedBy__c), @@ -18,6 +20,7 @@ BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsername__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsername__c.field-meta.xml index 983c511aa..62c398af8 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsername__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedByUsername__c.field-meta.xml @@ -1,10 +1,13 @@ LoggedByUsername__c + Active + PII false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedBy__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedBy__c.field-meta.xml index 4ec369f88..191a8e006 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedBy__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggedBy__c.field-meta.xml @@ -1,12 +1,14 @@ LoggedBy__c + Active SetNull false User Logs false + Confidential true false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggerVersionNumber__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggerVersionNumber__c.field-meta.xml index c7c2dc9f9..0b7be0a63 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggerVersionNumber__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoggerVersionNumber__c.field-meta.xml @@ -1,11 +1,12 @@ LoggerVersionNumber__c - false + Active true 14 false + Confidential true false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginApplication__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginApplication__c.field-meta.xml index 4bcd663cb..e42e11766 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginApplication__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginApplication__c.field-meta.xml @@ -1,10 +1,12 @@ LoginApplication__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginBrowser__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginBrowser__c.field-meta.xml index 2be71f897..97063d0b7 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginBrowser__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginBrowser__c.field-meta.xml @@ -1,10 +1,12 @@ LoginBrowser__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginDomain__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginDomain__c.field-meta.xml index 0a09ad39b..6329389d4 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginDomain__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginDomain__c.field-meta.xml @@ -1,9 +1,11 @@ LoginDomain__c + DeprecateCandidate false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginHistoryId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginHistoryId__c.field-meta.xml index 223808493..add18c405 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginHistoryId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginHistoryId__c.field-meta.xml @@ -1,10 +1,12 @@ LoginHistoryId__c + Active false 18 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginPlatform__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginPlatform__c.field-meta.xml index fa8e137b4..a9a29ca21 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginPlatform__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginPlatform__c.field-meta.xml @@ -1,10 +1,12 @@ LoginPlatform__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginType__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginType__c.field-meta.xml index a77abc6ed..57cbd20b6 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginType__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LoginType__c.field-meta.xml @@ -1,16 +1,18 @@ LoginType__c + Active false false + Confidential false false false Picklist - true + false Application false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/LogoutUrl__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogoutUrl__c.field-meta.xml index d425158db..f35ff6c5a 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/LogoutUrl__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/LogoutUrl__c.field-meta.xml @@ -1,9 +1,11 @@ LogoutUrl__c + Active false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/MaxLogEntryLoggingLevelOrdinal__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/MaxLogEntryLoggingLevelOrdinal__c.field-meta.xml index 3b9fd89f8..0fb475547 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/MaxLogEntryLoggingLevelOrdinal__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/MaxLogEntryLoggingLevelOrdinal__c.field-meta.xml @@ -1,10 +1,12 @@ MaxLogEntryLoggingLevelOrdinal__c + Active The highest logging level ordinal of any related log entries false The highest logging level ordinal of any related log entries + Confidential LogEntry__c.LoggingLevelOrdinal__c LogEntry__c.Log__c max diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkId__c.field-meta.xml index 4dce88d11..1b8b4704e 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkId__c.field-meta.xml @@ -1,12 +1,15 @@ NetworkId__c + Active + PII The Network ID of the Community user's site. Set with Network.getNetworkId() true The Network ID of the user's Community site. 18 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLoginUrl__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLoginUrl__c.field-meta.xml index 8947a42eb..44c7ec30f 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLoginUrl__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLoginUrl__c.field-meta.xml @@ -1,9 +1,12 @@ NetworkLoginUrl__c + Active + PII false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLogoutUrl__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLogoutUrl__c.field-meta.xml index 23ccc0c38..77a88f9af 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLogoutUrl__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkLogoutUrl__c.field-meta.xml @@ -1,9 +1,12 @@ NetworkLogoutUrl__c + Active + PII false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkName__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkName__c.field-meta.xml index c52ed9bbf..aad5b9d23 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkName__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkName__c.field-meta.xml @@ -1,12 +1,15 @@ NetworkName__c + Active + PII The name of the user's Community site (based on NetworkId). false The name of the user's Community site (based on NetworkId). 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkSelfRegistrationUrl__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkSelfRegistrationUrl__c.field-meta.xml index d5b60fcd4..d9b81b741 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkSelfRegistrationUrl__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkSelfRegistrationUrl__c.field-meta.xml @@ -1,9 +1,12 @@ NetworkSelfRegistrationUrl__c + Active + PII false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkUrlPathPrefix__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkUrlPathPrefix__c.field-meta.xml index 9cbee92bf..db0b54d42 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkUrlPathPrefix__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/NetworkUrlPathPrefix__c.field-meta.xml @@ -1,6 +1,8 @@ NetworkUrlPathPrefix__c + Active + PII The UrlPathPrefix is a unique string at the end of the URL for this community. For example, in the community URL CommunitiesSubdomainName.force.com/customers, customers is the UrlPathPrefix. false @@ -9,6 +11,7 @@ 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationDomainUrl__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationDomainUrl__c.field-meta.xml index cb516caf2..90353792c 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationDomainUrl__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationDomainUrl__c.field-meta.xml @@ -1,11 +1,13 @@ OrganizationDomainUrl__c + Active The value returned from Url..getOrgDomainUrl() false The value returned from Url..getOrgDomainUrl() false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationEnvironmentType__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationEnvironmentType__c.field-meta.xml index c97b39abf..e3161535d 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationEnvironmentType__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationEnvironmentType__c.field-meta.xml @@ -1,9 +1,11 @@ OrganizationEnvironmentType__c + Active false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationId__c.field-meta.xml index e4d6e4a80..2a96add22 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationId__c.field-meta.xml @@ -1,10 +1,12 @@ OrganizationId__c + Active false 18 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceName__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceName__c.field-meta.xml index c189ad52d..907db2c48 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceName__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceName__c.field-meta.xml @@ -1,10 +1,12 @@ OrganizationInstanceName__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml index b48a16f08..9d4fbd0ae 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml @@ -1,9 +1,11 @@ OrganizationInstanceReleaseCycle__c + DeprecateCandidate false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationName__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationName__c.field-meta.xml index ba640028b..c39b0f25a 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationName__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationName__c.field-meta.xml @@ -1,10 +1,12 @@ OrganizationName__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationNamespacePrefix__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationNamespacePrefix__c.field-meta.xml index 4ae2ead4c..9d2894868 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationNamespacePrefix__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationNamespacePrefix__c.field-meta.xml @@ -1,10 +1,12 @@ OrganizationNamespacePrefix__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationType__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationType__c.field-meta.xml index 7a01a4aa3..1d31de9c0 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationType__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/OrganizationType__c.field-meta.xml @@ -1,9 +1,11 @@ OrganizationType__c + Active false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ParentLog__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ParentLog__c.field-meta.xml index bb157f66a..60ac85829 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ParentLog__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ParentLog__c.field-meta.xml @@ -1,6 +1,7 @@ ParentLog__c + Active SetNull The log from the original transaction that initiated a child log - for example, batch jobs have start, execute and finish methods. All 3 are considered separate transactions. By using the parent log, logs from all 3 transactions can be linked together. @@ -12,6 +13,7 @@ Related Logs ChildLogs false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/Priority__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/Priority__c.field-meta.xml index ca1c247ce..ca874c767 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/Priority__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/Priority__c.field-meta.xml @@ -1,9 +1,11 @@ Priority__c + Active false false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileId__c.field-meta.xml index 1d26d7edb..edf5eccc1 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileId__c.field-meta.xml @@ -1,10 +1,13 @@ ProfileId__c + Active + PII false 18 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileLink__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileLink__c.field-meta.xml index b8ad1b45e..4f8d1b680 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileLink__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileLink__c.field-meta.xml @@ -1,11 +1,14 @@ ProfileLink__c + Active + PII false HYPERLINK('/' + ProfileId__c, ProfileName__c, '_top') BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileName__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileName__c.field-meta.xml index e1b6d776a..cbeac9be7 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileName__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ProfileName__c.field-meta.xml @@ -1,10 +1,13 @@ ProfileName__c + Active + PII false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/Scenario__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/Scenario__c.field-meta.xml index b405391b0..cd2b5f618 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/Scenario__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/Scenario__c.field-meta.xml @@ -1,11 +1,12 @@ Scenario__c - false + Active true 255 false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionId__c.field-meta.xml index 6af2db111..359321471 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionId__c.field-meta.xml @@ -1,10 +1,13 @@ SessionId__c + Active + PII false 120 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionSecurityLevel__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionSecurityLevel__c.field-meta.xml index d4ec41751..49e4c7a4d 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionSecurityLevel__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionSecurityLevel__c.field-meta.xml @@ -1,9 +1,12 @@ SessionSecurityLevel__c + Active + PII false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionType__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionType__c.field-meta.xml index e41afb24a..cdbb77849 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionType__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/SessionType__c.field-meta.xml @@ -1,16 +1,19 @@ SessionType__c + Active + PII false false + Confidential false false false Picklist - true + false Aura false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/SourceIp__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/SourceIp__c.field-meta.xml index f177ca234..a67bae0be 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/SourceIp__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/SourceIp__c.field-meta.xml @@ -1,10 +1,13 @@ SourceIp__c + Active + PII false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/StartTime__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/StartTime__c.field-meta.xml index 409ff58f2..616da84b8 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/StartTime__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/StartTime__c.field-meta.xml @@ -1,8 +1,10 @@ StartTime__c + Active false + Confidential LogEntry__c.Timestamp__c LogEntry__c.Log__c min diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/Status__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/Status__c.field-meta.xml index bf1cae059..eb11b8566 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/Status__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/Status__c.field-meta.xml @@ -1,9 +1,11 @@ Status__c + Active false false + Confidential true true false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemModeSummary__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemModeSummary__c.field-meta.xml index 3b18e2fc0..d6a066814 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemModeSummary__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemModeSummary__c.field-meta.xml @@ -1,11 +1,13 @@ SystemModeSummary__c + Active false 'API.' + TEXT(ApiVersion__c) + '.' + TEXT(SystemMode__c) BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemMode__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemMode__c.field-meta.xml index 9146f17f2..34f5cdfb5 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemMode__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/SystemMode__c.field-meta.xml @@ -1,16 +1,18 @@ SystemMode__c + Active false false + Confidential false false false Picklist - true + false ANONYMOUS false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/ThemeDisplayed__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/ThemeDisplayed__c.field-meta.xml index 69b64bc4c..f3fe03b69 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/ThemeDisplayed__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/ThemeDisplayed__c.field-meta.xml @@ -1,9 +1,11 @@ ThemeDisplayed__c + Active false false + Confidential true false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneId__c.field-meta.xml index ecb5aac42..eff2f6e20 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneId__c.field-meta.xml @@ -1,10 +1,13 @@ TimeZoneId__c + Active + PII false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneName__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneName__c.field-meta.xml index b8c9a602c..e1ed26fe0 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneName__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TimeZoneName__c.field-meta.xml @@ -1,10 +1,12 @@ TimeZoneName__c + Active false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalDEBUGLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalDEBUGLogEntries__c.field-meta.xml index f42654084..14f5cd9e6 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalDEBUGLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalDEBUGLogEntries__c.field-meta.xml @@ -1,10 +1,12 @@ TotalDEBUGLogEntries__c + Active The total number of log entries with logging level == 'DEBUG' false The total number of log entries with logging level == 'DEBUG' + Confidential LogEntry__c.LoggingLevel__c equals diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalERRORLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalERRORLogEntries__c.field-meta.xml index 09cd4ac86..464682efe 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalERRORLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalERRORLogEntries__c.field-meta.xml @@ -1,10 +1,12 @@ TotalERRORLogEntries__c + Active The total number of log entries with logging level == 'ERROR' false The total number of log entries with logging level == 'ERROR' + Confidential LogEntry__c.LoggingLevel__c equals diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINELogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINELogEntries__c.field-meta.xml index 0a4f026cf..e12c52688 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINELogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINELogEntries__c.field-meta.xml @@ -1,10 +1,12 @@ TotalFINELogEntries__c + Active The total number of log entries with logging level == 'FINE' false The total number of log entries with logging level == 'FINE' + Confidential LogEntry__c.LoggingLevel__c equals diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINERLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINERLogEntries__c.field-meta.xml index 91eedeb0d..1cbf43179 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINERLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINERLogEntries__c.field-meta.xml @@ -1,10 +1,12 @@ TotalFINERLogEntries__c + Active The total number of log entries with logging level == 'FINER' false The total number of log entries with logging level == 'FINER' + Confidential LogEntry__c.LoggingLevel__c equals diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINESTLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINESTLogEntries__c.field-meta.xml index 779b45b6e..164fffdb0 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINESTLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalFINESTLogEntries__c.field-meta.xml @@ -1,10 +1,12 @@ TotalFINESTLogEntries__c + Active The total number of log entries with logging level == 'FINEST' false The total number of log entries with logging level == 'FINEST' + Confidential LogEntry__c.LoggingLevel__c equals diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalINFOLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalINFOLogEntries__c.field-meta.xml index 02d97d7b6..80ed537f2 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalINFOLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalINFOLogEntries__c.field-meta.xml @@ -1,10 +1,12 @@ TotalINFOLogEntries__c + Active The total number of log entries with logging level == 'INFO' false The total number of log entries with logging level == 'INFO' + Confidential LogEntry__c.LoggingLevel__c equals diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLimitsCpuTimeUsed__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLimitsCpuTimeUsed__c.field-meta.xml index 972db8fee..9fca72780 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLimitsCpuTimeUsed__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLimitsCpuTimeUsed__c.field-meta.xml @@ -1,8 +1,10 @@ TotalLimitsCpuTimeUsed__c + Active false + Confidential LogEntry__c.LimitsCpuTimeUsed__c LogEntry__c.Log__c max diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLogEntries__c.field-meta.xml index ceca12d64..36d9deff2 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalLogEntries__c.field-meta.xml @@ -1,6 +1,7 @@ TotalLogEntries__c + Active false TotalERRORLogEntries__c + TotalWARNLogEntries__c @@ -14,6 +15,7 @@ 18 false 0 + Confidential false false Number diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalWARNLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalWARNLogEntries__c.field-meta.xml index 7666dd0b8..08bed31a2 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalWARNLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TotalWARNLogEntries__c.field-meta.xml @@ -1,10 +1,12 @@ TotalWARNLogEntries__c + Active The total number of log entries with logging level == 'WARN' false The total number of log entries with logging level == 'WARN' + Confidential LogEntry__c.LoggingLevel__c equals diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/TransactionId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/TransactionId__c.field-meta.xml index 7136a2d2a..662f4844c 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/TransactionId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/TransactionId__c.field-meta.xml @@ -1,11 +1,13 @@ TransactionId__c + Active false true 36 false + Confidential true false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseDefinitionKey__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseDefinitionKey__c.field-meta.xml index c54766a66..c2ae56674 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseDefinitionKey__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseDefinitionKey__c.field-meta.xml @@ -1,10 +1,13 @@ UserLicenseDefinitionKey__c + Active + PII https://developer.salesforce.com/docs/atlas.en-us.object_reference.meta/object_reference/sforce_api_objects_userlicense.htm false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseId__c.field-meta.xml index 9e92f3071..28a5ae14c 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseId__c.field-meta.xml @@ -1,10 +1,13 @@ UserLicenseId__c + Active + PII false 18 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseName__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseName__c.field-meta.xml index 7cf97606e..8f340098d 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseName__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLicenseName__c.field-meta.xml @@ -1,16 +1,19 @@ UserLicenseName__c + Active + PII false false + Confidential false false false Picklist - true + false Analytics Cloud Integration User false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevelOrdinal__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevelOrdinal__c.field-meta.xml index f1bcfd2ee..02ce36afd 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevelOrdinal__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevelOrdinal__c.field-meta.xml @@ -1,11 +1,14 @@ UserLoggingLevelOrdinal__c + Active + PII false 2 false 0 + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevel__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevel__c.field-meta.xml index 1dfe76c41..f8913eb9f 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevel__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserLoggingLevel__c.field-meta.xml @@ -1,9 +1,12 @@ UserLoggingLevel__c + Active + PII false false + Confidential true false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleId__c.field-meta.xml index 63c2ca897..f4756a88a 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleId__c.field-meta.xml @@ -1,10 +1,13 @@ UserRoleId__c + Active + PII false 18 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleLink__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleLink__c.field-meta.xml index 343de48a7..e0c23ebd7 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleLink__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleLink__c.field-meta.xml @@ -1,11 +1,14 @@ UserRoleLink__c + Active + PII false HYPERLINK('/' + UserRoleId__c, UserRoleName__c, '_top') BlankAsZero false + Confidential false false Text diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleName__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleName__c.field-meta.xml index aa92901ab..3c12f6f96 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleName__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserRoleName__c.field-meta.xml @@ -1,10 +1,13 @@ UserRoleName__c + Active + PII false 255 false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserType__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserType__c.field-meta.xml index ffa8936f1..0e257bf66 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/UserType__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/UserType__c.field-meta.xml @@ -1,9 +1,12 @@ UserType__c + Active + PII false false + Confidential false false false diff --git a/nebula-logger/core/main/log-management/objects/Log__c/fields/WasLoggedByCurrentUser__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/fields/WasLoggedByCurrentUser__c.field-meta.xml index 23ac754cf..7ce6d64de 100644 --- a/nebula-logger/core/main/log-management/objects/Log__c/fields/WasLoggedByCurrentUser__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/Log__c/fields/WasLoggedByCurrentUser__c.field-meta.xml @@ -1,10 +1,13 @@ WasLoggedByCurrentUser__c + Active + PII false $User.Id == LoggedBy__c BlankAsZero + Confidential false false Checkbox diff --git a/nebula-logger/core/main/log-management/objects/Log__c/listViews/LogsToPurgeSoon.listView-meta.xml b/nebula-logger/core/main/log-management/objects/Log__c/listViews/LogsToPurgeSoon.listView-meta.xml new file mode 100644 index 000000000..4b1e004ac --- /dev/null +++ b/nebula-logger/core/main/log-management/objects/Log__c/listViews/LogsToPurgeSoon.listView-meta.xml @@ -0,0 +1,25 @@ + + + LogsToPurgeSoon + NAME + LoggedByUsernameLink__c + TransactionId__c + OWNER.ALIAS + Priority__c + Status__c + Scenario__c + StartTime__c + LogPurgeAction__c + LogRetentionDate__c + Everything + + LogRetentionDate__c + notEqual + + + LogRetentionDate__c + lessOrEqual + NEXT_N_DAYS:10 + + + diff --git a/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/TotalLogEntries__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/TotalLogEntries__c.field-meta.xml index 57bdb6bc4..78325805e 100644 --- a/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/TotalLogEntries__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/TotalLogEntries__c.field-meta.xml @@ -1,8 +1,10 @@ TotalLogEntries__c + Active false + Confidential LogEntryTag__c.Tag__c count false diff --git a/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/UniqueId__c.field-meta.xml b/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/UniqueId__c.field-meta.xml index 426eb9778..af3b61747 100644 --- a/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/UniqueId__c.field-meta.xml +++ b/nebula-logger/core/main/log-management/objects/LoggerTag__c/fields/UniqueId__c.field-meta.xml @@ -1,6 +1,7 @@ UniqueId__c + Active false An external ID field used to ensure that Logger Tag records are unique true @@ -8,6 +9,7 @@ 255 false + Confidential false false false diff --git a/nebula-logger/core/main/configuration/permissionsets/LoggerAdmin.permissionset-meta.xml b/nebula-logger/core/main/log-management/permissionsets/LoggerAdmin.permissionset-meta.xml similarity index 94% rename from nebula-logger/core/main/configuration/permissionsets/LoggerAdmin.permissionset-meta.xml rename to nebula-logger/core/main/log-management/permissionsets/LoggerAdmin.permissionset-meta.xml index 994fb5e48..6cf000c60 100644 --- a/nebula-logger/core/main/configuration/permissionsets/LoggerAdmin.permissionset-meta.xml +++ b/nebula-logger/core/main/log-management/permissionsets/LoggerAdmin.permissionset-meta.xml @@ -164,6 +164,11 @@ LogEntry__c.ComponentType__c true + + false + LogEntry__c.DatabaseResultCollectionSize__c + true + false LogEntry__c.DatabaseResultCollectionType__c @@ -289,6 +294,51 @@ LogEntry__c.HasStackTrace__c true + + false + LogEntry__c.HttpRequestBody__c + true + + + false + LogEntry__c.HttpRequestBodyMasked__c + true + + + false + LogEntry__c.HttpRequestEndpoint__c + true + + + false + LogEntry__c.HttpRequestMethod__c + true + + + false + LogEntry__c.HttpResponseBody__c + true + + + false + LogEntry__c.HttpResponseBodyMasked__c + true + + + false + LogEntry__c.HttpResponseHeaderKeys__c + true + + + false + LogEntry__c.HttpResponseStatusCode__c + true + + + false + LogEntry__c.HttpResponseStatus__c + true + false LogEntry__c.LimitsAggregateQueriesMax__c @@ -584,6 +634,11 @@ LogEntry__c.Origin__c true + + false + LogEntry__c.RecordCollectionSize__c + true + false LogEntry__c.RecordCollectionType__c @@ -729,6 +784,11 @@ Log__c.LogEntriesSummary__c true + + true + Log__c.LogPurgeAction__c + true + true Log__c.LogRetentionDate__c diff --git a/nebula-logger/core/main/configuration/permissionsets/LoggerEndUser.permissionset-meta.xml b/nebula-logger/core/main/log-management/permissionsets/LoggerEndUser.permissionset-meta.xml similarity index 98% rename from nebula-logger/core/main/configuration/permissionsets/LoggerEndUser.permissionset-meta.xml rename to nebula-logger/core/main/log-management/permissionsets/LoggerEndUser.permissionset-meta.xml index e5e391d44..6ba95179a 100644 --- a/nebula-logger/core/main/configuration/permissionsets/LoggerEndUser.permissionset-meta.xml +++ b/nebula-logger/core/main/log-management/permissionsets/LoggerEndUser.permissionset-meta.xml @@ -54,6 +54,11 @@ LogEntry__c.ComponentType__c true + + false + LogEntry__c.DatabaseResultCollectionSize__c + true + false LogEntry__c.DatabaseResultCollectionType__c @@ -464,6 +469,11 @@ LogEntry__c.Origin__c true + + false + LogEntry__c.RecordCollectionSize__c + true + false LogEntry__c.RecordCollectionType__c @@ -481,12 +491,12 @@ false - LogEntry__c.RecordJson__c + LogEntry__c.RecordJsonMasked__c true false - LogEntry__c.RecordJsonMasked__c + LogEntry__c.RecordJson__c true @@ -574,6 +584,11 @@ Log__c.LogEntriesSummary__c true + + false + Log__c.LogPurgeAction__c + true + false Log__c.LogRetentionDate__c diff --git a/nebula-logger/core/main/configuration/permissionsets/LoggerLogCreator.permissionset-meta.xml b/nebula-logger/core/main/log-management/permissionsets/LoggerLogCreator.permissionset-meta.xml similarity index 100% rename from nebula-logger/core/main/configuration/permissionsets/LoggerLogCreator.permissionset-meta.xml rename to nebula-logger/core/main/log-management/permissionsets/LoggerLogCreator.permissionset-meta.xml diff --git a/nebula-logger/core/main/configuration/permissionsets/LoggerLogViewer.permissionset-meta.xml b/nebula-logger/core/main/log-management/permissionsets/LoggerLogViewer.permissionset-meta.xml similarity index 94% rename from nebula-logger/core/main/configuration/permissionsets/LoggerLogViewer.permissionset-meta.xml rename to nebula-logger/core/main/log-management/permissionsets/LoggerLogViewer.permissionset-meta.xml index d538daa7a..02a039642 100644 --- a/nebula-logger/core/main/configuration/permissionsets/LoggerLogViewer.permissionset-meta.xml +++ b/nebula-logger/core/main/log-management/permissionsets/LoggerLogViewer.permissionset-meta.xml @@ -92,6 +92,11 @@ LogEntry__c.ComponentType__c true + + false + LogEntry__c.DatabaseResultCollectionSize__c + true + false LogEntry__c.DatabaseResultCollectionType__c @@ -217,6 +222,51 @@ LogEntry__c.HasStackTrace__c true + + false + LogEntry__c.HttpRequestBody__c + true + + + false + LogEntry__c.HttpRequestBodyMasked__c + true + + + false + LogEntry__c.HttpRequestEndpoint__c + true + + + false + LogEntry__c.HttpRequestMethod__c + true + + + false + LogEntry__c.HttpResponseBody__c + true + + + false + LogEntry__c.HttpResponseBodyMasked__c + true + + + false + LogEntry__c.HttpResponseHeaderKeys__c + true + + + false + LogEntry__c.HttpResponseStatusCode__c + true + + + false + LogEntry__c.HttpResponseStatus__c + true + false LogEntry__c.LimitsAggregateQueriesMax__c @@ -512,6 +562,11 @@ LogEntry__c.Origin__c true + + false + LogEntry__c.RecordCollectionSize__c + true + false LogEntry__c.RecordCollectionType__c @@ -529,12 +584,12 @@ false - LogEntry__c.RecordJson__c + LogEntry__c.RecordJsonMasked__c true false - LogEntry__c.RecordJsonMasked__c + LogEntry__c.RecordJson__c true @@ -657,6 +712,11 @@ Log__c.LogEntriesSummary__c true + + false + Log__c.LogPurgeAction__c + true + false Log__c.LogRetentionDate__c diff --git a/nebula-logger/core/main/log-management/quickActions/Log__c.Manage.quickAction-meta.xml b/nebula-logger/core/main/log-management/quickActions/Log__c.Manage.quickAction-meta.xml index cab796550..35958bec4 100644 --- a/nebula-logger/core/main/log-management/quickActions/Log__c.Manage.quickAction-meta.xml +++ b/nebula-logger/core/main/log-management/quickActions/Log__c.Manage.quickAction-meta.xml @@ -35,6 +35,11 @@ LogRetentionDate__c Edit + + false + LogPurgeAction__c + Edit + diff --git a/nebula-logger/core/main/log-management/triggers/Log.trigger b/nebula-logger/core/main/log-management/triggers/Log.trigger index e7bfb5ad4..f0f150e19 100644 --- a/nebula-logger/core/main/log-management/triggers/Log.trigger +++ b/nebula-logger/core/main/log-management/triggers/Log.trigger @@ -3,5 +3,5 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// trigger Log on Log__c(before insert, before update, before delete, after insert, after update, after delete, after undelete) { - new LogHandler().execute(); + LoggerSObjectHandler.getHandler(Schema.Log__c.SObjectType, new LogHandler()).execute(); } diff --git a/nebula-logger/core/main/log-management/triggers/LogEntry.trigger b/nebula-logger/core/main/log-management/triggers/LogEntry.trigger index 07009e45a..9399171e3 100644 --- a/nebula-logger/core/main/log-management/triggers/LogEntry.trigger +++ b/nebula-logger/core/main/log-management/triggers/LogEntry.trigger @@ -3,5 +3,5 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// trigger LogEntry on LogEntry__c(before insert, before update, before delete, after insert, after update, after delete, after undelete) { - new LogEntryHandler().execute(); + LoggerSObjectHandler.getHandler(Schema.LogEntry__c.SObjectType, new LogEntryHandler()).execute(); } diff --git a/nebula-logger/core/main/log-management/triggers/LogEntryEvent.trigger b/nebula-logger/core/main/log-management/triggers/LogEntryEvent.trigger index b3dbb6d23..9c35ec4d6 100644 --- a/nebula-logger/core/main/log-management/triggers/LogEntryEvent.trigger +++ b/nebula-logger/core/main/log-management/triggers/LogEntryEvent.trigger @@ -3,5 +3,5 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// trigger LogEntryEvent on LogEntryEvent__e(after insert) { - new LogEntryEventHandler().execute(); + LoggerSObjectHandler.getHandler(Schema.LogEntryEvent__e.SObjectType, new LogEntryEventHandler()).execute(); } diff --git a/nebula-logger/core/main/log-management/triggers/LogEntryTag.trigger b/nebula-logger/core/main/log-management/triggers/LogEntryTag.trigger index ef2855e3e..a1fd400ab 100644 --- a/nebula-logger/core/main/log-management/triggers/LogEntryTag.trigger +++ b/nebula-logger/core/main/log-management/triggers/LogEntryTag.trigger @@ -3,5 +3,5 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// trigger LogEntryTag on LogEntryTag__c(before insert, before update, before delete, after insert, after update, after delete, after undelete) { - new LogEntryTagHandler().execute(); + LoggerSObjectHandler.getHandler(Schema.LogEntryTag__c.SObjectType, new LogEntryTagHandler()).execute(); } diff --git a/nebula-logger/core/main/log-management/triggers/LoggerTag.trigger b/nebula-logger/core/main/log-management/triggers/LoggerTag.trigger index 150848217..44427188a 100644 --- a/nebula-logger/core/main/log-management/triggers/LoggerTag.trigger +++ b/nebula-logger/core/main/log-management/triggers/LoggerTag.trigger @@ -3,5 +3,5 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// trigger LoggerTag on LoggerTag__c(before insert, before update, before delete, after insert, after update, after delete, after undelete) { - new LoggerTagHandler().execute(); + LoggerSObjectHandler.getHandler(Schema.LoggerTag__c.SObjectType, new LoggerTagHandler()).execute(); } diff --git a/nebula-logger/core/main/logger-engine/classes/ComponentLogger.cls b/nebula-logger/core/main/logger-engine/classes/ComponentLogger.cls index bf84cb5de..6e6623317 100644 --- a/nebula-logger/core/main/logger-engine/classes/ComponentLogger.cls +++ b/nebula-logger/core/main/logger-engine/classes/ComponentLogger.cls @@ -52,7 +52,10 @@ public inherited sharing class ComponentLogger { Logger.saveLog(saveMethod); return Logger.getTransactionId(); } catch (Exception apexException) { - throw new AuraHandledException(apexException.getMessage()); + String errorMessage = apexException.getMessage() + '\n' + apexException.getStackTraceString(); + AuraHandledException auraException = new AuraHandledException(errorMessage); + auraException.setMessage(errorMessage); + throw auraException; } } diff --git a/nebula-logger/core/main/logger-engine/classes/FlowCollectionLogEntry.cls b/nebula-logger/core/main/logger-engine/classes/FlowCollectionLogEntry.cls index f0219cbba..b0111382b 100644 --- a/nebula-logger/core/main/logger-engine/classes/FlowCollectionLogEntry.cls +++ b/nebula-logger/core/main/logger-engine/classes/FlowCollectionLogEntry.cls @@ -88,13 +88,16 @@ global inherited sharing class FlowCollectionLogEntry { @InvocableMethod( category='Logging' label='Add Log Entry for an SObject Record Collection' - description='Creates a log entry for a flow or process builder and stores the record as JSON' + description='Creates a log entry for a flow or process builder and stores the record list as JSON' ) global static List addFlowCollectionEntries(List flowCollectionLogEntries) { List shadowLogEntries = new List(); for (FlowCollectionLogEntry flowCollectionLogEntry : flowCollectionLogEntries) { FlowLogger.LogEntry shadowLogEntry = (FlowLogger.LogEntry) JSON.deserialize(JSON.serialize(flowCollectionLogEntry), FlowLogger.LogEntry.class); - shadowLogEntry.addToLoggerBuffer()?.setRecord(flowCollectionLogEntry.records); + LogEntryEventBuilder builder = shadowLogEntry.addToLoggerBuffer()?.setRecord(flowCollectionLogEntry.records); + if (builder.shouldSave() == true && flowCollectionLogEntry.records?.size() > 0) { + builder.getLogEntryEvent().RecordSObjectType__c = flowCollectionLogEntry.records.get(0).getSObjectType().getDescribe().getName(); + } shadowLogEntries.add(shadowLogEntry); } diff --git a/nebula-logger/core/main/logger-engine/classes/LogEntryEventBuilder.cls b/nebula-logger/core/main/logger-engine/classes/LogEntryEventBuilder.cls index ae8f8cc06..efeb9b18e 100644 --- a/nebula-logger/core/main/logger-engine/classes/LogEntryEventBuilder.cls +++ b/nebula-logger/core/main/logger-engine/classes/LogEntryEventBuilder.cls @@ -9,7 +9,7 @@ * @see Logger */ @SuppressWarnings( - 'PMD.NcssTypeCount, PMD.ApexCRUDViolation, PMD.AvoidGlobalModifier, PMD.PropertyNamingConventions, PMD.CognitiveComplexity, PMD.StdCyclomaticComplexity, PMD.CyclomaticComplexity' + 'PMD.ApexCRUDViolation, PMD.AvoidGlobalModifier, PMD.CognitiveComplexity, PMD.CyclomaticComplexity, PMD.ExcessiveClassLength, PMD.NcssTypeCount, PMD.PropertyNamingConventions, PMD.StdCyclomaticComplexity' ) global with sharing class LogEntryEventBuilder { private static final String API_VERSION = getApiVersion(); @@ -71,7 +71,7 @@ global with sharing class LogEntryEventBuilder { set; } - private static final List CACHED_DATA_MASK_RULES { + private static final Map CACHED_DATA_MASK_RULES { get { if (CACHED_DATA_MASK_RULES == null) { CACHED_DATA_MASK_RULES = loadDataMaskRules(); @@ -167,7 +167,7 @@ global with sharing class LogEntryEventBuilder { global LogEntryEventBuilder setMessage(String message) { // To help with debugging unit tests, always run System.debug statement in a test context if ((this.shouldSave == true || System.Test.isRunningTest() == true) && this.logEntryEvent != null) { - String cleanedMessage = applyDataMaskRules(message, true, false); + String cleanedMessage = applyDataMaskRules(message); Boolean messageMasked = cleanedMessage != message; String truncatedMessage = truncateFieldValue(Schema.LogEntryEvent__e.Message__c, cleanedMessage); Boolean messageTruncated = @@ -210,15 +210,16 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(Database.DeleteResult deleteResult) { - if (this.shouldSave == false || deleteResult == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = 1; this.logEntryEvent.DatabaseResultCollectionType__c = 'Single'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(deleteResult)); this.logEntryEvent.DatabaseResultType__c = Database.DeleteResult.class.getName(); - return this.setRecordId(deleteResult.getId()); + return this.setRecordId(deleteResult?.getId()); } /** @@ -227,15 +228,16 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(Database.MergeResult mergeResult) { - if (this.shouldSave == false || mergeResult == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = 1; this.logEntryEvent.DatabaseResultCollectionType__c = 'Single'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(mergeResult)); this.logEntryEvent.DatabaseResultType__c = Database.MergeResult.class.getName(); - return this.setRecordId(mergeResult.getId()); + return this.setRecordId(mergeResult?.getId()); } /** @@ -244,15 +246,16 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(Database.SaveResult saveResult) { - if (this.shouldSave == false || saveResult == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = 1; this.logEntryEvent.DatabaseResultCollectionType__c = 'Single'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(saveResult)); this.logEntryEvent.DatabaseResultType__c = Database.SaveResult.class.getName(); - return this.setRecordId(saveResult.getId()); + return this.setRecordId(saveResult?.getId()); } /** @@ -261,18 +264,19 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(Database.UpsertResult upsertResult) { - if (this.shouldSave == false || upsertResult == null) { + if (this.shouldSave == false) { return this; } // Upsert has 2 subtypes (surprise!) - insert and update - so, UpsertResult has an extra method to take into account String subtype = upsertResult.isCreated() ? 'Insert' : 'Update'; + this.logEntryEvent.DatabaseResultCollectionSize__c = 1; this.logEntryEvent.DatabaseResultCollectionType__c = 'Single'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(upsertResult)); this.logEntryEvent.DatabaseResultType__c = Database.UpsertResult.class.getName() + '.' + subtype; - return this.setRecordId(upsertResult.getId()); + return this.setRecordId(upsertResult?.getId()); } /** @@ -281,15 +285,16 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(Database.UndeleteResult undeleteResult) { - if (this.shouldSave == false || undeleteResult == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = 1; this.logEntryEvent.DatabaseResultCollectionType__c = 'Single'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(undeleteResult)); this.logEntryEvent.DatabaseResultType__c = Database.UndeleteResult.class.getName(); - return this.setRecordId(undeleteResult.getId()); + return this.setRecordId(undeleteResult?.getId()); } /** @@ -298,10 +303,11 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(List deleteResults) { - if (this.shouldSave == false || deleteResults == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = deleteResults?.size(); this.logEntryEvent.DatabaseResultCollectionType__c = 'List'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(deleteResults)); this.logEntryEvent.DatabaseResultType__c = Database.DeleteResult.class.getName(); @@ -315,10 +321,11 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(List mergeResults) { - if (this.shouldSave == false || mergeResults == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = mergeResults?.size(); this.logEntryEvent.DatabaseResultCollectionType__c = 'List'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(mergeResults)); this.logEntryEvent.DatabaseResultType__c = Database.MergeResult.class.getName(); @@ -332,10 +339,11 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(List saveResults) { - if (this.shouldSave == false || saveResults == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = saveResults?.size(); this.logEntryEvent.DatabaseResultCollectionType__c = 'List'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(saveResults)); this.logEntryEvent.DatabaseResultType__c = Database.SaveResult.class.getName(); @@ -349,10 +357,11 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(List upsertResults) { - if (this.shouldSave == false || upsertResults == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = upsertResults?.size(); this.logEntryEvent.DatabaseResultCollectionType__c = 'List'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(upsertResults)); this.logEntryEvent.DatabaseResultType__c = Database.UpsertResult.class.getName(); @@ -366,10 +375,11 @@ global with sharing class LogEntryEventBuilder { * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods */ global LogEntryEventBuilder setDatabaseResult(List undeleteResults) { - if (this.shouldSave == false || undeleteResults == null) { + if (this.shouldSave == false) { return this; } + this.logEntryEvent.DatabaseResultCollectionSize__c = undeleteResults?.size(); this.logEntryEvent.DatabaseResultCollectionType__c = 'List'; this.logEntryEvent.DatabaseResultJson__c = truncateFieldValue(Schema.LogEntryEvent__e.DatabaseResultJson__c, JSON.serializePretty(undeleteResults)); this.logEntryEvent.DatabaseResultType__c = Database.UndeleteResult.class.getName(); @@ -405,6 +415,7 @@ global with sharing class LogEntryEventBuilder { return this; } + this.logEntryEvent.RecordCollectionSize__c = 1; this.logEntryEvent.RecordCollectionType__c = 'Single'; this.logEntryEvent.RecordId__c = recordId; this.logEntryEvent.RecordSObjectClassification__c = getSObjectClassification(recordId.getSObjectType()); @@ -422,6 +433,7 @@ global with sharing class LogEntryEventBuilder { return this; } + this.logEntryEvent.RecordCollectionSize__c = 1; this.logEntryEvent.RecordCollectionType__c = 'Single'; String recordJson = getJson(record); @@ -430,7 +442,7 @@ global with sharing class LogEntryEventBuilder { this.logEntryEvent.RecordSObjectClassification__c = 'Unknown'; this.logEntryEvent.RecordSObjectType__c = 'Unknown'; } else { - String cleanedRecordJson = applyDataMaskRules(recordJson, false, true); + String cleanedRecordJson = applyDataMaskRules(recordJson); Boolean recordJsonMasked = cleanedRecordJson != recordJson; this.logEntryEvent.RecordId__c = record.Id; @@ -454,23 +466,17 @@ global with sharing class LogEntryEventBuilder { return this; } + this.logEntryEvent.RecordCollectionSize__c = records?.size(); this.logEntryEvent.RecordCollectionType__c = 'List'; - // First, try to use the list's SObject Type - if the list is null or an instance of List, it won't have an SObject Type - // If using the list doesn't work, and there are 1+ records in the list, then try to use the first record's SObject Type - if it's null, it won't have an SObject Type - // If the first object doesn't work, then just log as is a generic SObject - no need to waste additional CPU time on it Schema.SObjectType sobjectType = records?.getSObjectType(); - if (sobjectType == null && records != null && records.size() >= 1) { - sobjectType = records.get(0)?.getSObjectType(); - } - String recordJson = getJson(records); if (sobjectType == null) { this.logEntryEvent.RecordJson__c = recordJson; this.logEntryEvent.RecordSObjectClassification__c = 'Unknown'; this.logEntryEvent.RecordSObjectType__c = 'Unknown'; } else { - String cleanedRecordJson = applyDataMaskRules(recordJson, false, true); + String cleanedRecordJson = applyDataMaskRules(recordJson); Boolean recordJsonMasked = cleanedRecordJson != recordJson; this.logEntryEvent.RecordJson__c = cleanedRecordJson; @@ -483,6 +489,48 @@ global with sharing class LogEntryEventBuilder { return this; } + /** + * @description Sets the log entry event's HTTP Request fields + * @param request The instance of `HttpRequest` to log + * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods + */ + global LogEntryEventBuilder setHttpRequestDetails(HttpRequest request) { + if (this.shouldSave == false || request == null) { + return this; + } + + String cleanedRequestBody = applyDataMaskRules(request.getBody()); + Boolean requestBodyMasked = cleanedRequestBody != request.getBody(); + + this.logEntryEvent.HttpRequestBody__c = cleanedRequestBody; + this.logEntryEvent.HttpRequestBodyMasked__c = requestBodyMasked; + this.logEntryEvent.HttpRequestCompressed__c = request.getCompressed(); + this.logEntryEvent.HttpRequestEndpoint__c = request.getEndpoint(); + this.logEntryEvent.HttpRequestMethod__c = request.getMethod(); + return this; + } + + /** + * @description Sets the log entry event's HTTP Response fields + * @param response The instance of `HttpResponse` to log + * @return The same instance of `LogEntryEventBuilder`, useful for chaining methods + */ + global LogEntryEventBuilder setHttpResponseDetails(HttpResponse response) { + if (this.shouldSave == false || response == null) { + return this; + } + + String cleanedResponseBody = applyDataMaskRules(response.getBody()); + Boolean responseBodyMasked = cleanedResponseBody != response.getBody(); + + this.logEntryEvent.HttpResponseBody__c = cleanedResponseBody; + this.logEntryEvent.HttpResponseBodyMasked__c = responseBodyMasked; + this.logEntryEvent.HttpResponseHeaderKeys__c = String.join(response.getHeaderKeys(), '\n'); + this.logEntryEvent.HttpResponseStatus__c = response.getStatus(); + this.logEntryEvent.HttpResponseStatusCode__c = response.getStatusCode(); + return this; + } + /** * @description Appends the tag to the existing list of tags * @param tag The string to use as a tag for the current entry @@ -541,9 +589,14 @@ global with sharing class LogEntryEventBuilder { } List stackTraceLines = new List(); + String previousStackTraceLine; for (String currentStackTraceLine : stackTraceString.split('\n')) { - // Don't include the logging system's classes in the stack trace + // Duplicate lines are sometimes introduced, so skip the current line if it's the same as the previous line + if (currentStackTraceLine == previousStackTraceLine) { + continue; + } + // Don't include the logging system's classes in the stack trace Boolean ignoreLine = false; for (String ignoredClass : IGNORED_CLASSES) { if (currentStackTraceLine.contains('.' + ignoredClass + '.')) { @@ -557,6 +610,7 @@ global with sharing class LogEntryEventBuilder { } if (ignoreLine == false) { + previousStackTraceLine = currentStackTraceLine; stackTraceLines.add(currentStackTraceLine); } } @@ -669,7 +723,9 @@ global with sharing class LogEntryEventBuilder { return; } - this.logEntryEvent.Tags__c = String.escapeSingleQuotes(String.join(new List(this.tags), '\n')); + List sortedTags = new List(this.tags); + sortedTags.sort(); + this.logEntryEvent.Tags__c = String.escapeSingleQuotes(String.join(sortedTags, '\n')); } private void setTransactionDetails() { @@ -739,17 +795,17 @@ global with sharing class LogEntryEventBuilder { // Private static helper methods @TestVisible - private static void addMockDataMaskRule(LogEntryDataMaskRule__mdt dataMaskRule) { - CACHED_DATA_MASK_RULES.add(dataMaskRule); + private static void setMockDataMaskRule(LogEntryDataMaskRule__mdt dataMaskRule) { + CACHED_DATA_MASK_RULES.put(dataMaskRule.DeveloperName, dataMaskRule); } - private static String applyDataMaskRules(String dataInput, Boolean applyToMessage, Boolean applyToRecordJson) { + private static String applyDataMaskRules(String dataInput) { if (Logger.getUserSettings().IsDataMaskingEnabled__c == false || String.isBlank(dataInput) == true) { return dataInput; } - for (LogEntryDataMaskRule__mdt dataMaskRule : CACHED_DATA_MASK_RULES) { - if ((applyToMessage && dataMaskRule.ApplyToMessage__c) == true || (applyToRecordJson && dataMaskRule.ApplyToRecordJson__c) == true) { + for (LogEntryDataMaskRule__mdt dataMaskRule : CACHED_DATA_MASK_RULES.values()) { + if (dataMaskRule.IsEnabled__c == true) { dataInput = dataInput.replaceAll(dataMaskRule.SensitiveDataRegEx__c, dataMaskRule.ReplacementRegEx__c); } } @@ -766,16 +822,16 @@ global with sharing class LogEntryEventBuilder { return userJson.substringAfter('/data/').substringBefore('/sobjects/User'); } - private static List loadDataMaskRules() { - List activeDataMaskRules = new List(); + private static Map loadDataMaskRules() { + Map activeDataMaskRules = new Map(); for (LogEntryDataMaskRule__mdt dataMaskRule : LogEntryDataMaskRule__mdt.getAll().values()) { if (dataMaskRule.IsEnabled__c == true) { - activeDataMaskRules.add(dataMaskRule); + activeDataMaskRules.put(dataMaskRule.DeveloperName, dataMaskRule); } } if (System.Test.isRunningTest() == true) { // Tests shouldn't rely on the actual CMDT rules in the org - // Clear the org's loaded records during tests, and mock via addMockDataMaskRule() + // Clear the org's loaded records during tests, and mock via setMockDataMaskRule() activeDataMaskRules.clear(); } return activeDataMaskRules; @@ -938,7 +994,7 @@ global with sharing class LogEntryEventBuilder { private static SObject queryNetwork() { Id networkId = Network.getNetworkId(); - if (networkId == null || Schema.getGlobalDescribe().containsKey('Network') == false) { + if (networkId == null || Type.forName('Network') == null) { return null; } diff --git a/nebula-logger/core/main/logger-engine/classes/Logger.cls b/nebula-logger/core/main/logger-engine/classes/Logger.cls index 8781cc8bc..c602a5566 100644 --- a/nebula-logger/core/main/logger-engine/classes/Logger.cls +++ b/nebula-logger/core/main/logger-engine/classes/Logger.cls @@ -15,27 +15,29 @@ global with sharing class Logger { // There's no reliable way to get the version number dynamically in Apex @TestVisible - private static final String CURRENT_VERSION_NUMBER = 'v4.7.0'; + private static final String CURRENT_VERSION_NUMBER = 'v4.7.1'; private static final LoggingLevel DEFAULT_LOGGING_LEVEL = LoggingLevel.DEBUG; private static final List LOG_ENTRIES_BUFFER = new List(); - @TestVisible - private static final Map SCENARIO_TO_MOCK_SCENARIO_RULE = new Map(); + private static final Map MOCK_SCENARIO_TO_SCENARIO_RULE = new Map(); private static final String TRANSACTION_ID = setTransactionId(); private static final Quiddity TRANSACTION_QUIDDITY = setTransactionQuiddity(); private static Integer currentTransactionEntryNumber = 1; + @TestVisible + private static String lastSaveMethodNameUsed; private static String parentLogTransactionId; - private static String transactionScenario = getUserSettings().DefaultLogScenario__c; + @TestVisible + private static Integer saveLogCallCount = 0; private static Boolean suspendSaving = false; + private static String transactionScenario = getUserSettings().DefaultLogScenario__c; private static LoggerSettings__c userSettings; - @TestVisible - private static SaveMethod transactionSaveMethod { + private static String transactionSaveMethodName { get { - if (transactionSaveMethod == null) { - transactionSaveMethod = getDefaultSaveMethod(); + if (transactionSaveMethodName == null) { + transactionSaveMethodName = getUserSettings().DefaultSaveMethod__c; } - return transactionSaveMethod; + return transactionSaveMethodName; } set; } @@ -51,7 +53,7 @@ global with sharing class Logger { } static { - System.debug(LoggingLevel.INFO, 'Logger - Version Number: ' + CURRENT_VERSION_NUMBER); + System.debug(LoggingLevel.INFO, 'Logger - Version Number: ' + getVersionNumber()); System.debug(LoggingLevel.INFO, 'Logger - Transaction ID: ' + getTransactionId()); } @@ -65,6 +67,25 @@ global with sharing class Logger { SYNCHRONOUS_DML } + // System info methods + + /** + * @description Returns the current version number of Nebula Logger + * @return The current version number, in the format `v0.0.0` + */ + public static String getVersionNumber() { + return CURRENT_VERSION_NUMBER; + } + + /** + * @description Returns the current namespace of Nebula Logger + * @return The current namespace prefix, or an empty string when no namespace is being used + */ + public static String getNamespacePrefix() { + String className = Logger.class.getName(); + return className.contains('.') ? className.substringBefore('.') : ''; + } + // Settings management methods /** @@ -2474,7 +2495,7 @@ global with sharing class Logger { } if (System.Test.isRunningTest() == true) { - matchingScenarioRule = SCENARIO_TO_MOCK_SCENARIO_RULE.get(scenario); + matchingScenarioRule = MOCK_SCENARIO_TO_SCENARIO_RULE.get(scenario); } if (matchingScenarioRule != null) { @@ -2484,7 +2505,7 @@ global with sharing class Logger { @TestVisible private static void setMockScenarioRule(LogScenarioRule__mdt scenarioRule) { - SCENARIO_TO_MOCK_SCENARIO_RULE.put(scenarioRule.Scenario__c, scenarioRule); + MOCK_SCENARIO_TO_SCENARIO_RULE.put(scenarioRule.Scenario__c, scenarioRule); } /** @@ -2492,7 +2513,7 @@ global with sharing class Logger { * @return The enum value of Logger.SaveMethod to use for any calls to saveLog() in the current transaction */ global static SaveMethod getSaveMethod() { - return transactionSaveMethod; + return getSaveMethod(transactionSaveMethodName); } /** @@ -2500,7 +2521,7 @@ global with sharing class Logger { * @param saveMethod The enum value of Logger.SaveMethod to use for any other calls to saveLog() in the current transaction */ global static void setSaveMethod(SaveMethod saveMethod) { - transactionSaveMethod = saveMethod; + transactionSaveMethodName = saveMethod.name(); } /** @@ -2508,7 +2529,7 @@ global with sharing class Logger { */ @InvocableMethod(category='Logging' label='Save Log' description='Saves any log entries that have been generated') global static void saveLog() { - saveLog(transactionSaveMethod); + saveLog(transactionSaveMethodName); } /** @@ -2516,8 +2537,19 @@ global with sharing class Logger { * All subsequent calls to saveLog() will use the transaction save method. * @param saveMethod The enum value of Logger.SaveMethod to use for this specific save action. */ - @SuppressWarnings('PMD.NcssMethodCount') global static void saveLog(SaveMethod saveMethod) { + saveLog(saveMethod.name()); + } + + /** + * @description Saves any entries in Logger's buffer, using the specified save method for only this call. + * All subsequent calls to saveLog() will use the transaction save method. + * @param saveMethodName The String value of the save method to use for this specific save action. + */ + @SuppressWarnings('PMD.NcssMethodCount') + public static void saveLog(String saveMethodName) { + saveLogCallCount++; + lastSaveMethodNameUsed = saveMethodName; if (LOG_ENTRIES_BUFFER.isEmpty() || getUserSettings().IsSavingEnabled__c == false) { return; } @@ -2531,7 +2563,7 @@ global with sharing class Logger { } if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { - finest(getSavingLogSystemMessage(saveMethod)); + finest(getSavingLogSystemMessage(saveMethodName)); } List logEntryEvents = new List(); @@ -2553,20 +2585,25 @@ global with sharing class Logger { // This gives us a chance to run the handler class & handler plugins before insert, // allowing the plugins to make further changes to the `LogEntryEvent__e` records // So, execute the handler, which internally then executes any plugins - new LogEntryEventHandler().executeBeforeInsert(logEntryEvents); + LoggerTriggerableContext logEntryEventsBeforeContext = new LoggerTriggerableContext( + Schema.LogEntryEvent__e.SObjectType, + TriggerOperation.BEFORE_INSERT, + logEntryEvents + ); + LoggerSObjectHandler.getHandler(Schema.LogEntryEvent__e.SObjectType)?.overrideTriggerableContext(logEntryEventsBeforeContext).execute(); // Now that the plugins have run, double check to make sure that saving should still happen if (getBufferSize() == 0) { return; } - switch on saveMethod { + switch on getSaveMethod(saveMethodName) { when EVENT_BUS { - List saveResults = EventBus.publish(logEntryEvents); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntryEvent__e.SObjectType, saveResults); + List saveResults = LoggerDataStore.getEventBus().publishRecords(logEntryEvents); + LoggerEmailSender.sendErrorEmail(Schema.LogEntryEvent__e.SObjectType, saveResults); } when QUEUEABLE { - System.enqueueJob(new QueueableSaver(logEntryEvents)); + LoggerDataStore.getJobQueue().enqueueJob(new QueueableSaver(logEntryEvents)); } when REST { // If the user doesn't have a session ID (e.g., site guest user), the REST API call will fail @@ -2578,7 +2615,12 @@ global with sharing class Logger { } } when SYNCHRONOUS_DML { - new LogEntryEventHandler().executeAfterInsert(logEntryEvents); + LoggerTriggerableContext logEntryEventsAfterContext = new LoggerTriggerableContext( + Schema.LogEntryEvent__e.SObjectType, + TriggerOperation.AFTER_INSERT, + logEntryEvents + ); + LoggerSObjectHandler.getHandler(Schema.LogEntryEvent__e.SObjectType)?.overrideTriggerableContext(logEntryEventsAfterContext).execute(); } } @@ -2593,42 +2635,18 @@ global with sharing class Logger { * @return The matching instance of LoggingLevel (or a default value if a match is not found) */ global static LoggingLevel getLoggingLevel(String loggingLevelName) { - if (loggingLevelName != null) { - loggingLevelName = loggingLevelName.trim().toUpperCase(); - } - - switch on loggingLevelName { - when 'NONE' { - return LoggingLevel.NONE; - } - when 'ERROR' { - return LoggingLevel.ERROR; - } - when 'WARN' { - return LoggingLevel.WARN; - } - when 'INFO' { - return LoggingLevel.INFO; - } - when 'DEBUG' { - return LoggingLevel.DEBUG; - } - when 'FINE' { - return LoggingLevel.FINE; - } - when 'FINER' { - return LoggingLevel.FINER; - } - when 'FINEST' { - return LoggingLevel.FINEST; - } - when else { - if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { - String message = 'Logger - Unknown logging level ' + loggingLevelName + 'specified, using ' + DEFAULT_LOGGING_LEVEL.name(); - finest(message); - } - return DEFAULT_LOGGING_LEVEL; + try { + return System.LoggingLevel.valueOf(loggingLevelName?.trim().toUpperCase()); + } catch (NoSuchElementException ex) { + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + LogMessage logMessage = new LogMessage( + 'Logger - Unknown logging level {0} specified, using {1}', + loggingLevelName, + DEFAULT_LOGGING_LEVEL.name() + ); + finest(logMessage); } + return DEFAULT_LOGGING_LEVEL; } } @@ -2728,33 +2746,19 @@ global with sharing class Logger { } } - private static SaveMethod getDefaultSaveMethod() { - SaveMethod defaultSaveMethod; - - String saveMethodName = getUserSettings().DefaultSaveMethod__c?.toUpperCase(); - switch on saveMethodName { - when 'QUEUEABLE' { - defaultSaveMethod = SaveMethod.QUEUEABLE; - } - when 'REST' { - defaultSaveMethod = SaveMethod.REST; - } - when 'SYNCHRONOUS_DML' { - defaultSaveMethod = SaveMethod.SYNCHRONOUS_DML; - } - when else { - defaultSaveMethod = SaveMethod.EVENT_BUS; - } + private static SaveMethod getSaveMethod(String saveMethodName) { + try { + return SaveMethod.valueOf(saveMethodName); + } catch (NoSuchElementException ex) { + return null; } - - return defaultSaveMethod; } private static String getSuspendSavingLogSystemMessage() { return 'Logger - Logging suspended, ignoring call to saveLog()'; } - private static String getSavingLogSystemMessage(SaveMethod saveMethod) { + private static String getSavingLogSystemMessage(String saveMethodName) { String savingLogMessage = 'Logger - Saving ' + LOG_ENTRIES_BUFFER.size() + ' log entries via '; switch on getCurrentQuiddity() { when ANONYMOUS { @@ -2788,7 +2792,12 @@ global with sharing class Logger { savingLogMessage += getCurrentQuiddity()?.name(); } } - savingLogMessage += ', save method is ' + saveMethod.name(); + savingLogMessage += ', save method is ' + saveMethodName; + savingLogMessage += + '. Logger.saveLog() has been called ' + + saveLogCallCount + + (saveLogCallCount == 1 ? ' time' : ' times') + + ' in the current transaction.'; return savingLogMessage; } @@ -2805,12 +2814,11 @@ global with sharing class Logger { } /** - * @description Required by the Queueable interface, this method contains the logic executed when - * the current instance of the queue runs. - * @param queueableContext The context of the current queue. + * @description Asynchronoulsy publishes the list of `LogEntryEvent__e` records + * @param queueableContext The context of the current queue, provided by the platform */ global void execute(System.QueueableContext queueableContext) { - EventBus.publish(this.logEntryEvents); + LoggerDataStore.getEventBus().publishRecords(this.logEntryEvents); } } diff --git a/nebula-logger/core/main/logger-engine/classes/LoggerDataStore.cls b/nebula-logger/core/main/logger-engine/classes/LoggerDataStore.cls new file mode 100644 index 000000000..623230546 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/classes/LoggerDataStore.cls @@ -0,0 +1,295 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Logger Engine + * @description Class used to manage any data-related operations, including database DML statements, + * publishing platform events via the event bus, and enqueueing queueable jobs + */ +@SuppressWarnings('PMD.ApexCRUDViolation, PMD.CyclomaticComplexity, PMD.EmptyStatementBlock, PMD.ExcessivePublicCount, PMD.FieldDeclarationsShouldBeAtStart') +public without sharing class LoggerDataStore { + private static Database databaseInstance = new Database(); + private static EventBus eventBusInstance = new EventBus(); + private static JobQueue jobQueueInstance = new JobQueue(); + /** + * @description The instance `LoggerDataStore.Database` used for any DML + * operations in the current transaction. + * @return The singleton instance of `LoggerDataStore.Database` + */ + public static Database getDatabase() { + return databaseInstance; + } + + /** + * @description The instance `LoggerDataStore.EventBus` used for publishing + * platform events in the current transaction. + * @return The singleton instance of `LoggerDataStore.EventBus` + */ + public static EventBus getEventBus() { + return eventBusInstance; + } + + /** + * @description The instance `LoggerDataStore.JobQueue` used for enqueuing + * any queueable jobs in the current transaction. + * @return The singleton instance of `LoggerDataStore.JobQueue` + */ + public static JobQueue getJobQueue() { + return jobQueueInstance; + } + + @TestVisible + private static void setMock(Database mockDatabase) { + databaseInstance = mockDatabase; + } + + @TestVisible + private static void setMock(EventBus mockEventBus) { + eventBusInstance = mockEventBus; + } + + @TestVisible + private static void setMock(JobQueue mockJobQueue) { + jobQueueInstance = mockJobQueue; + } + + /** + * @description Class used to centralize the handling of any DML operations + */ + public virtual class Database { + protected Database() { + } + + /** + * @description Executes a `delete` DML operation for the `SObject` record + * @param record The `SObject` record to delete + * @return The instance of `Database.DeleteResult`, generated by the platform when deleting the record + */ + public virtual Database.DeleteResult deleteRecord(SObject record) { + return System.Database.delete(record); + } + + /** + * @description Executes a `delete` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to delete + * @return The instance of `List`, generated by the platform when deleting the records + */ + public virtual List deleteRecords(List records) { + return System.Database.delete(records); + } + + /** + * @description Executes a `delete` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to delete + * @param allOrNone Controls if all records must be deleted (`true`), or if partial deletion should be used (`false`) + * @return The instance of `List`, generated by the platform when deleting the records + */ + public virtual List deleteRecords(List records, Boolean allOrNone) { + return System.Database.delete(records, allOrNone); + } + + /** + * @description Executes a `delete` DML operation for the `SObject` record, followed by hard deleting the + * record using `Database.emptyRecycleBin(record)` + * @param record The `SObject` record to delete + * @return The instance of `Database.DeleteResult`, generated by the platform when deleting the record + */ + public virtual Database.DeleteResult hardDeleteRecord(SObject record) { + return this.hardDeleteRecords(new List{ record }).get(0); + } + + /** + * @description Executes a `delete` DML operation on the provided list of `SObject` records, followed by hard deleting the + * records using `Database.emptyRecycleBin(records)` + * @param records The list of `SObject` records to delete + * @return The instance of `List`, generated by the platform when deleting the records + */ + public virtual List hardDeleteRecords(List records) { + List results = this.deleteRecords(records); + if (records.isEmpty() == false) { + System.Database.emptyRecycleBin(records); + } + return results; + } + + /** + * @description Executes an `insert` DML operation on the provided `SObject` record + * @param record The `SObject` record to insert + * @return The instance of `Database.SaveResult`, generated by the platform when creating the record + */ + public virtual Database.SaveResult insertRecord(SObject record) { + return System.Database.insert(record); + } + + /** + * @description Executes an `insert` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to insert + * @return The instance of `List`, generated by the platform when creating the records + */ + public virtual List insertRecords(List records) { + return System.Database.insert(records); + } + + /** + * @description Executes an `insert` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to insert + * @param allOrNone Controls if all records must be created (`true`), or if partial creation should be used (`false`) + * @return The instance of `List`, generated by the platform when creating the records + */ + public virtual List insertRecords(List records, Boolean allOrNone) { + return System.Database.insert(records, allOrNone); + } + + /** + * @description Executes an `insert` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to insert + * @param dmlOptions Controls additional DML options, using the provided instance of `Database.DmlOptions` + * @return The instance of `List`, generated by the platform when creating the records + */ + public virtual List insertRecords(List records, Database.DmlOptions dmlOptions) { + return System.Database.insert(records, dmlOptions); + } + + /** + * @description Executes an `undelete` DML operation on the provided `SObject` record + * @param record The `SObject` record to undelete + * @return The instance of `Database.UndeleteResult`, generated by the platform when undeleting the record + */ + public virtual Database.UndeleteResult undeleteRecord(SObject record) { + return System.Database.undelete(record); + } + + /** + * @description Executes an `undelete` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to undelete + * @return The instance of `List`, generated by the platform when deleting the records + */ + public virtual List undeleteRecords(List records) { + return System.Database.undelete(records); + } + + /** + * @description Executes an `undelete` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to undelete + * @param allOrNone Controls if all records must be undeleted (`true`), or if partial undeletion should be used (`false`) + * @return The instance of `List`, generated by the platform when undeleting the records + */ + public virtual List undeleteRecords(List records, Boolean allOrNone) { + return System.Database.undelete(records, allOrNone); + } + + /** + * @description Executes an `update` DML operation on the provided `SObject` record + * @param record The `SObject` record to update + * @return The instance of `Database.SaveResult`, generated by the platform when updating the record + */ + public virtual Database.SaveResult updateRecord(SObject record) { + return System.Database.update(record); + } + + /** + * @description Executes an `update` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to update + * @return The instance of `List`, generated by the platform when updating the records + */ + public virtual List updateRecords(List records) { + return System.Database.update(records); + } + + /** + * @description Executes an `update` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to update + * @param allOrNone Controls if all records must be updated (`true`), or if partial updates should be used (`false`) + * @return The instance of `List`, generated by the platform when updating the records + */ + public virtual List updateRecords(List records, Boolean allOrNone) { + return System.Database.update(records, allOrNone); + } + + /** + * @description Executes an `update` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to update + * @param dmlOptions Controls additional DML options, using the provided instance of `Database.DmlOptions` + * @return The instance of `List`, generated by the platform when updating the records + */ + public virtual List updateRecords(List records, Database.DmlOptions dmlOptions) { + return System.Database.update(records, dmlOptions); + } + + /** + * @description Executes an `upsert` DML operation on the provided list of `SObject` records + * @param record The `SObject` record to update + * @param externalIdField The `SObjectField` of the external ID field on the target `SObject` to use for upserting + * @return The instance of `Database.UpsertResult`, generated by the platform when upserting the record + */ + public virtual Database.UpsertResult upsertRecord(SObject record, Schema.SObjectField externalIdField) { + return System.Database.upsert(record, externalIdField); + } + + /** + * @description Executes an `upsert` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to upsert + * @param externalIdField The `SObjectField` of the external ID field on the target `SObject` to use for upserting + * @return The instance of `List`, generated by the platform when Upserting the records + */ + public virtual List upsertRecords(List records, Schema.SObjectField externalIdField) { + return System.Database.upsert(records, externalIdField); + } + + /** + * @description Executes an `upsert` DML operation on the provided list of `SObject` records + * @param records The list of `SObject` records to upsert + * @param externalIdField The `SObjectField` of the external ID field on the target `SObject` to use for upserting + * @param allOrNone Controls if all records must be updated (`true`), or if partial updates should be used (`false`) + * @return The instance of `List`, generated by the platform when Upserting the records + */ + public virtual List upsertRecords(List records, Schema.SObjectField externalIdField, Boolean allOrNone) { + return System.Database.upsert(records, externalIdField, allOrNone); + } + } + + /** + * @description Class used to centralize the handling of any platform event publishing operations + */ + public virtual class EventBus { + protected EventBus() { + } + + /** + * @description Publishes a single platform event record, using `EventBus.publish(SObject record); + * @param platformEvent The platform event record to publish + * @return The instance of `Database.SaveResult`, generated by the platform when publishing the platform event record + */ + public virtual Database.SaveResult publishRecord(SObject platformEvent) { + return System.EventBus.publish(platformEvent); + } + + /** + * @description Publishes a list of platform event records, using `EventBus.publish(List records); + * @param platformEvents The list of platform event records to publish + * @return The instance of `List`, generated by the platform when publishing the platform event records + */ + public virtual List publishRecords(List platformEvents) { + return System.EventBus.publish(platformEvents); + } + } + + /** + * @description Class used to centralize the handling of enqueueing any queueable jobs + */ + public virtual class JobQueue { + protected JobQueue() { + } + + /** + * @description Enqueues a queueable job to execute asynchronously, using `System.enqueueJob(Queueable queueableJob)` + * @param queueableJob An instance of a `Queueable` class that should be enqueued + * @return The `Id` of the queueable job + */ + public virtual Id enqueueJob(Queueable queueableJob) { + return System.enqueueJob(queueableJob); + } + } +} diff --git a/nebula-logger/core/tests/common/classes/LoggerTestUtils.cls-meta.xml b/nebula-logger/core/main/logger-engine/classes/LoggerDataStore.cls-meta.xml similarity index 100% rename from nebula-logger/core/tests/common/classes/LoggerTestUtils.cls-meta.xml rename to nebula-logger/core/main/logger-engine/classes/LoggerDataStore.cls-meta.xml diff --git a/nebula-logger/core/main/logger-engine/classes/LoggerSObjectHandler.cls b/nebula-logger/core/main/logger-engine/classes/LoggerSObjectHandler.cls new file mode 100644 index 000000000..878ac6d6e --- /dev/null +++ b/nebula-logger/core/main/logger-engine/classes/LoggerSObjectHandler.cls @@ -0,0 +1,287 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Logger Engine + * @description Abstract class used by trigger handlers for shared logic + */ +@SuppressWarnings('PMD.ApexCRUDViolation, PMD.CyclomaticComplexity, PMD.EmptyStatementBlock, PMD.FieldDeclarationsShouldBeAtStart') +public without sharing abstract class LoggerSObjectHandler { + private static Boolean isEnabled = true; + private static final Map> SOBJECT_TYPE_TO_EXECUTED_HANDLERS = new Map>(); + private static final Map SOBJECT_TYPE_TO_HANDLER_CONFIGURATIONS = queryHandlerConfigurations(); + + // Instance variables for top-level class + @TestVisible + protected LoggerTriggerableContext input { + get { + if (input == null) { + input = new LoggerTriggerableContext(this.getSObjectType(), this.triggerOperationType, this.triggerNew, this.triggerNewMap, this.triggerOldMap); + } + return input; + } + set; + } + + @TestVisible + private TriggerOperation triggerOperationType; + @TestVisible + private List triggerNew; + @TestVisible + private Map triggerNewMap; + @TestVisible + private Map triggerOldMap; + @TestVisible + private LoggerSObjectHandler__mdt handlerConfiguration; + private List pluginConfigurations = new List(); + private List executedApexPlugins = new List(); + private List executedFlowPlugins = new List(); + + /** + * @description Returns an instance of `LoggerSObjectHandler` that has been built & configured for the specified `SObjectType` + * @param sobjectType The instance `SObjectType` to check for a configured instance of `LoggerSObjectHandler` + * @return The Apex class that extends `LoggerSObjectHandler` and has been configured for the specified `SObjectType` + */ + public static LoggerSObjectHandler getHandler(Schema.SObjectType sobjectType) { + return getHandler(sobjectType, null); + } + + /** + * @description Returns an instance of `LoggerSObjectHandler` that has been built & configured for the specified `SObjectType` + * @param sobjectType The instance `SObjectType` to check for a configured instance of `LoggerSObjectHandler` + * @param defaultImplementation A default implementation of `LoggerSObjectHandler` that should be used, if a configuration cannot be found `LoggerSObjectHandler__mdt` + * @return The Apex class that extends `LoggerSObjectHandler` and has been configured for the specified `SObjectType` + */ + public static LoggerSObjectHandler getHandler(Schema.SObjectType sobjectType, LoggerSObjectHandler defaultImplementation) { + String handlerApexClassName = getHandlerConfiguration(sobjectType)?.SObjectHandlerApexClass__c; + // TODO need to handle namespace prefix for Apex classes + return handlerApexClassName == null ? defaultImplementation : (LoggerSObjectHandler) Type.forName(handlerApexClassName)?.newInstance(); + } + + /** + * @description Default constructor + */ + public LoggerSObjectHandler() { + this.triggerOperationType = Trigger.operationType; + this.triggerNew = Trigger.new; + this.triggerNewMap = Trigger.newMap; + this.triggerOldMap = Trigger.oldMap; + this.handlerConfiguration = SOBJECT_TYPE_TO_HANDLER_CONFIGURATIONS.get(this.getSObjectType()); + this.pluginConfigurations = LoggerPlugin.getFilteredPluginConfigurations( + new List{ Schema.LoggerPlugin__mdt.SObjectHandlerApexClass__c, Schema.LoggerPlugin__mdt.SObjectHandlerFlowName__c }, + Schema.LoggerPlugin__mdt.SObjectHandlerExecutionOrder__c + ); + } + + /** + * @description Returns the SObject Type that the handler is responsible for processing + * @return The instance of `SObjectType` + */ + public abstract Schema.SObjectType getSObjectType(); + + /** + * @description Provides the ability to override the instance `LoggerTriggerableContext` that is normally + * provided internally by `LoggerSObjectHandler` + * @param input The instance of `LoggerTriggerableContext` to use within the trigger handler class + * @return The same instance of `LoggerSObjectHandler`, useful for chaining methods + */ + public LoggerSObjectHandler overrideTriggerableContext(LoggerTriggerableContext input) { + this.input = input; + return this; + } + + /** + * @description Runs the handler class's logic, as well as any configured plugins + */ + public void execute() { + if (System.Test.isRunningTest() == true && SOBJECT_TYPE_TO_EXECUTED_HANDLERS.containsKey(this.getSObjectType()) == false) { + SOBJECT_TYPE_TO_EXECUTED_HANDLERS.put(this.getSObjectType(), new List()); + } + + Boolean shouldExecute = isEnabled && (this.handlerConfiguration == null || this.handlerConfiguration.IsEnabled__c == true); + if (shouldExecute == false) { + return; + } + + switch on this.input.triggerOperationType { + when BEFORE_INSERT { + this.executeBeforeInsert(this.input.triggerNew); + } + when BEFORE_UPDATE { + this.executeBeforeUpdate(this.input.triggerNewMap, this.input.triggerOldMap); + } + when BEFORE_DELETE { + this.executeBeforeDelete(this.input.triggerNewMap); + } + when AFTER_INSERT { + // Platform Events don't have an ID field, thus Trigger.newMap doesn't work for LogEntryEvent__e + // For custom objects, Map is more convenient since it provides both the keys & values + // 2 AFTER_INSERT methods are used here in the framework, with the expectation that only 1 will be implemented per handler class + this.executeAfterInsert(this.input.triggerNew); + this.executeAfterInsert(this.input.triggerNewMap); + } + when AFTER_UPDATE { + this.executeAfterUpdate(this.input.triggerNewMap, this.input.triggerOldMap); + } + when AFTER_DELETE { + this.executeAfterDelete(this.input.triggerNewMap); + } + when AFTER_UNDELETE { + this.executeAfterUndelete(this.input.triggerNewMap); + } + } + + this.executePlugins(); + + if (System.Test.isRunningTest() == true) { + SOBJECT_TYPE_TO_EXECUTED_HANDLERS.get(this.getSObjectType()).add(this); + } + } + + protected virtual void executeBeforeInsert(List triggerNew) { + } + + protected virtual void executeBeforeUpdate(Map triggerNewMap, Map triggerOldMap) { + } + + protected virtual void executeBeforeDelete(Map triggerNewMap) { + } + + // executeAfterInsert(List triggerNew) { + } + + protected virtual void executeAfterInsert(Map triggerNewMap) { + } + + protected virtual void executeAfterUpdate(Map triggerNewMap, Map triggerOldMap) { + } + + protected virtual void executeAfterDelete(Map triggerNewMap) { + } + + protected virtual void executeAfterUndelete(Map triggerNewMap) { + } + + private void executePlugins() { + if (this.pluginConfigurations == null || this.pluginConfigurations.isEmpty() == true) { + return; + } + + for (LoggerPlugin__mdt pluginConfiguration : this.pluginConfigurations) { + if (String.isNotBlank(pluginConfiguration.SObjectHandlerApexClass__c)) { + this.executeApexPlugin(pluginConfiguration); + } + if (String.isNotBlank(pluginConfiguration.SObjectHandlerFlowName__c)) { + this.executeFlowPlugin(pluginConfiguration); + } + } + } + + @SuppressWarnings('PMD.AvoidDebugStatements') + private void executeApexPlugin(LoggerPlugin__mdt configuration) { + try { + LoggerPlugin.Triggerable apexPlugin = LoggerPlugin.newTriggerableInstance(configuration.SObjectHandlerApexClass__c); + apexPlugin?.execute(configuration, this.input); + + if (System.Test.isRunningTest() == true && apexPlugin != null) { + this.executedApexPlugins.add(apexPlugin); + } + } catch (TypeException ex) { + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.warn('Unknown Apex class ' + configuration.SObjectHandlerApexClass__c + ', skipping plugin execution', configuration, ex); + } + } + } + + @SuppressWarnings('PMD.AvoidDebugStatements') + private void executeFlowPlugin(LoggerPlugin__mdt configuration) { + try { + Map flowInputs = new Map(); + flowInputs.put('pluginConfiguration', configuration); + flowInputs.put('pluginInput', this.input); + + Flow.Interview flowPlugin = Flow.Interview.createInterview(configuration.SObjectHandlerFlowName__c, flowInputs); + flowPlugin.start(); + + List updatedTriggerNew = (List) flowPlugin.getVariableValue('updatedTriggerNew'); + if (updatedTriggerNew != null && updatedTriggerNew.size() == this.input.triggerNew.size()) { + this.input.triggerNew.clear(); + this.input.triggerNew.addAll(updatedTriggerNew); + if (this.input.triggerNewMap != null) { + this.input.triggerNewMap.clear(); + this.input.triggerNewMap.putAll(new Map(updatedTriggerNew)); + } + } + if (System.Test.isRunningTest() == true && flowPlugin != null) { + this.executedFlowPlugins.add(flowPlugin); + } + } catch (TypeException ex) { + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.warn('Unknown Flow ' + configuration.SObjectHandlerFlowName__c + ', skipping plugin execution', configuration, ex); + } + } + } + + // Instance test helper methods, used for unit tests + @TestVisible + private List getPluginConfigurations() { + return this.pluginConfigurations; + } + + @TestVisible + private List getExecutedApexPlugins() { + return this.executedApexPlugins; + } + + @TestVisible + private List getExecutedFlowPlugins() { + return this.executedFlowPlugins; + } + + // Static test helper methods, used for integration tests + @TestVisible + private static Map> getExecutedHandlers() { + return SOBJECT_TYPE_TO_EXECUTED_HANDLERS; + } + + private static Map queryHandlerConfigurations() { + Map sobjectTypeToHandlerConfiguration = new Map(); + for (LoggerSObjectHandler__mdt handlerConfiguration : [ + SELECT IsEnabled__c, SObjectHandlerApexClass__c, SObjectType__r.QualifiedApiName + FROM LoggerSObjectHandler__mdt + WHERE IsEnabled__c = TRUE + ]) { + handlerConfiguration.SObjectType__c = handlerConfiguration.SObjectType__r.QualifiedApiName; + Schema.SObjectType sobjectType = ((SObject) Type.forName(handlerConfiguration.SObjectType__c).newInstance()).getSObjectType(); + sobjectTypeToHandlerConfiguration.put(sobjectType, handlerConfiguration); + } + + if (System.Test.isRunningTest() == true) { + sobjectTypeToHandlerConfiguration.clear(); + } + + return sobjectTypeToHandlerConfiguration; + } + + @TestVisible + private static LoggerSObjectHandler__mdt getHandlerConfiguration(Schema.SObjectType sobjectType) { + return SOBJECT_TYPE_TO_HANDLER_CONFIGURATIONS.get(sobjectType); + } + + @TestVisible + private static void setMock(LoggerSObjectHandler__mdt handlerConfiguration) { + //TODO cleanup code duplication + Schema.SObjectType sobjectType = ((SObject) Type.forName(handlerConfiguration.SObjectType__c).newInstance()).getSObjectType(); + SOBJECT_TYPE_TO_HANDLER_CONFIGURATIONS.put(sobjectType, handlerConfiguration); + } + + @TestVisible + private static void shouldExecute(Boolean shouldExecute) { + // Only used for testing purposes so tests can disable triggers to speed up + // DML statements that don't rely on trigger logic to run + isEnabled = shouldExecute; + } +} diff --git a/nebula-logger/core/tests/configuration/classes/LoggerEmailUtils_Tests.cls-meta.xml b/nebula-logger/core/main/logger-engine/classes/LoggerSObjectHandler.cls-meta.xml similarity index 100% rename from nebula-logger/core/tests/configuration/classes/LoggerEmailUtils_Tests.cls-meta.xml rename to nebula-logger/core/main/logger-engine/classes/LoggerSObjectHandler.cls-meta.xml diff --git a/nebula-logger/core/main/logger-engine/classes/LoggerTriggerableContext.cls b/nebula-logger/core/main/logger-engine/classes/LoggerTriggerableContext.cls new file mode 100644 index 000000000..da870609f --- /dev/null +++ b/nebula-logger/core/main/logger-engine/classes/LoggerTriggerableContext.cls @@ -0,0 +1,87 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Logger Engine + * @description Class used by the logging system for trigger contextual details + * @see LoggerSObjectHandler + * @see LoggerPlugin + */ +@SuppressWarnings('PMD.ApexDoc, PMD.ExcessiveParameterList') +public without sharing class LoggerTriggerableContext { + public Schema.SObjectType sobjectType { get; private set; } + public TriggerOperation triggerOperationType { get; private set; } + public List triggerNew { get; private set; } + public Map triggerNewMap { get; private set; } + public Map triggerOldMap { get; private set; } + + @AuraEnabled + public String sobjectTypeName { get; private set; } + + @AuraEnabled + public String triggerOperationTypeName { get; private set; } + + @AuraEnabled + public List triggerRecords { get; private set; } + + public LoggerTriggerableContext(Schema.SObjectType sobjectType, TriggerOperation triggerOperationType, List triggerNew) { + this(sobjectType, triggerOperationType, triggerNew, null, null); + } + + public LoggerTriggerableContext( + Schema.SObjectType sobjectType, + TriggerOperation triggerOperationType, + List triggerNew, + Map triggerNewMap, + Map triggerOldMap + ) { + // Trigger variables for Apex Developers + this.sobjectType = sobjectType; + this.triggerOperationType = triggerOperationType; + this.triggerNew = triggerNew; + this.triggerNewMap = triggerNewMap; + this.triggerOldMap = triggerOldMap; + + this.setupAdditionalProperties(); + } + + private void setupAdditionalProperties() { + // Additional invocable variables for Flow Builders (and Apex Developers too, if they want to use them) + this.sobjectTypeName = this.sobjectType.getDescribe().getName(); + this.triggerOperationTypeName = this.triggerOperationType?.name(); + this.triggerRecords = new List(); + if (this.triggerNew != null) { + for (SObject record : this.triggerNew) { + LoggerTriggerableContext.RecordInput recordInput = new LoggerTriggerableContext.RecordInput(); + recordInput.triggerRecordNew = record; + if (this.triggerOldMap != null && record.Id != null) { + recordInput.triggerRecordOld = this.triggerOldMap.get(record.Id); + } + + this.triggerRecords.add(recordInput); + } + } else if (this.triggerOldMap != null) { + for (SObject record : this.triggerOldMap.values()) { + LoggerTriggerableContext.RecordInput recordInput = new LoggerTriggerableContext.RecordInput(); + recordInput.triggerRecordOld = record; + + this.triggerRecords.add(recordInput); + } + } + } + + /** + * @description Class used by the logging system to provide trigger record details to Flow + */ + public class RecordInput { + @AuraEnabled + public SObject triggerRecordNew { get; private set; } + + @AuraEnabled + public SObject triggerRecordOld { get; private set; } + } + + // TODO add a constructor & move logic from LoggerSObjectHandler --> no longer a DTO? +} diff --git a/nebula-logger/core/tests/plugin-framework/classes/LoggerSObjectHandlerPlugin_Tests.cls-meta.xml b/nebula-logger/core/main/logger-engine/classes/LoggerTriggerableContext.cls-meta.xml similarity index 100% rename from nebula-logger/core/tests/plugin-framework/classes/LoggerSObjectHandlerPlugin_Tests.cls-meta.xml rename to nebula-logger/core/main/logger-engine/classes/LoggerTriggerableContext.cls-meta.xml diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/DatabaseResultCollectionSize__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/DatabaseResultCollectionSize__c.field-meta.xml new file mode 100644 index 000000000..3198c288d --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/DatabaseResultCollectionSize__c.field-meta.xml @@ -0,0 +1,15 @@ + + + DatabaseResultCollectionSize__c + The number of items contained in the collection of database results + false + false + false + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBodyMasked__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBodyMasked__c.field-meta.xml new file mode 100644 index 000000000..7e5541ea5 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBodyMasked__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpRequestBodyMasked__c + false + false + false + false + false + + Checkbox + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBody__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBody__c.field-meta.xml new file mode 100644 index 000000000..50414a97a --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestBody__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpRequestBody__c + false + + 131072 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestCompressed__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestCompressed__c.field-meta.xml new file mode 100644 index 000000000..ebf8828f6 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestCompressed__c.field-meta.xml @@ -0,0 +1,10 @@ + + + HttpRequestCompressed__c + false + false + + false + false + Checkbox + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestEndpoint__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestEndpoint__c.field-meta.xml new file mode 100644 index 000000000..f50bbbd77 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestEndpoint__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpRequestEndpoint__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestMethod__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestMethod__c.field-meta.xml new file mode 100644 index 000000000..2523f3b51 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpRequestMethod__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpRequestMethod__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBodyMasked__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBodyMasked__c.field-meta.xml new file mode 100644 index 000000000..b4d22a354 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBodyMasked__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseBodyMasked__c + false + false + false + false + false + + Checkbox + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBody__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBody__c.field-meta.xml new file mode 100644 index 000000000..b0f892ac1 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseBody__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseBody__c + false + + 131072 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseHeaderKeys__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseHeaderKeys__c.field-meta.xml new file mode 100644 index 000000000..d5ca1263f --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseHeaderKeys__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseHeaderKeys__c + false + + 1000 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatusCode__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatusCode__c.field-meta.xml new file mode 100644 index 000000000..d3f8431d7 --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatusCode__c.field-meta.xml @@ -0,0 +1,13 @@ + + + HttpResponseStatusCode__c + false + + 10 + false + 0 + false + false + Number + false + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatus__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatus__c.field-meta.xml new file mode 100644 index 000000000..a12ff395e --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/HttpResponseStatus__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpResponseStatus__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/LoginDomain__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/LoginDomain__c.field-meta.xml index 29706601a..50ba6cd74 100644 --- a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/LoginDomain__c.field-meta.xml +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/LoginDomain__c.field-meta.xml @@ -1,6 +1,7 @@ LoginDomain__c + DeprecateCandidate false false false diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/RecordCollectionSize__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/RecordCollectionSize__c.field-meta.xml new file mode 100644 index 000000000..1440e138d --- /dev/null +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/RecordCollectionSize__c.field-meta.xml @@ -0,0 +1,15 @@ + + + RecordCollectionSize__c + The number of items contained in the collection of database results + false + false + false + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/Topics__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/Topics__c.field-meta.xml index b83ece151..bea48807e 100644 --- a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/Topics__c.field-meta.xml +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/Topics__c.field-meta.xml @@ -1,6 +1,7 @@ Topics__c + DeprecateCandidate false false false diff --git a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/TransactionEntryNumber__c.field-meta.xml b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/TransactionEntryNumber__c.field-meta.xml index 1a57a3a12..c2e58be1b 100644 --- a/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/TransactionEntryNumber__c.field-meta.xml +++ b/nebula-logger/core/main/logger-engine/objects/LogEntryEvent__e/fields/TransactionEntryNumber__c.field-meta.xml @@ -6,7 +6,7 @@ false false false - + 10 true 0 diff --git a/nebula-logger/core/main/plugin-framework/classes/LoggerSObjectHandlerPlugin.cls b/nebula-logger/core/main/plugin-framework/classes/LoggerSObjectHandlerPlugin.cls deleted file mode 100644 index 6be6df44a..000000000 --- a/nebula-logger/core/main/plugin-framework/classes/LoggerSObjectHandlerPlugin.cls +++ /dev/null @@ -1,41 +0,0 @@ -//------------------------------------------------------------------------------------------------// -// This file is part of the Nebula Logger project, released under the MIT License. // -// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // -//------------------------------------------------------------------------------------------------// - -/** - * @group Plugin Framework - * @description Abstract class used to create custom Apex plugins to execute for all trigger operations on `Log__c` or `LogEntry__c` - */ -// TODO: If possible, convert to `global` for the managed package - there are some additional platform limitations -// and restrictions with making future changes to abstract classes & interfaces within managed packages -// which could lead to a lot of problems, so for now/indefinitely, plugins will only work in the unlocked package -public abstract class LoggerSObjectHandlerPlugin { - /** - * @description All instances of `LoggerSObjectHandlerPlugin` are dynamically created, which requires a parameterless constructor - */ - @SuppressWarnings('PMD.EmptyStatementBlock') - public LoggerSObjectHandlerPlugin() { - } - - /** - * @description This method is the entry point for plugins to execute any custom logic. - * It is automatically called by the logging system for any enabled plugins. - * Several trigger-based parameters are provided - these parameters should be used by plugins, - * instead of calling the platform's static variables directly - * (e.g., use the provided `triggerNew` variable instead of using `Trigger.new` directly, and so on). - * @param triggerOperationType The enum instance of `Trigger.operationType` at the time that the handler class is created - * @param triggerNew The value `Trigger.new` at the time that the handler class is created - * @param triggerNewMap The value `Trigger.newMap` at the time that the handler class is created - * @param triggerOld The value `Trigger.old` at the time that the handler class is created - * @param triggerOldMap The value `Trigger.oldMap` at the time that the handler class is created - */ - @SuppressWarnings('PMD.ExcessiveParameterList') - public abstract void execute( - TriggerOperation triggerOperationType, - List triggerNew, - Map triggerNewMap, - List triggerOld, - Map triggerOldMap - ); -} diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml b/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml deleted file mode 100644 index 32c04aa55..000000000 --- a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/fields/PluginApiName__c.field-meta.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - PluginApiName__c - The API name of the metadata to run, based on the Plugin Type - -Apex: The name of an Apex class that implements the interface LoggerSObjectHandlerPlugin - -Flow: The API name of a Flow that supports these inputs: - - triggerOperationType: text variable - - triggerNew: record collection variable - - triggerOld: record collection variable - false - DeveloperControlled - The API name of the metadata to run, based on the Plugin Type - -Apex: The name of an Apex class that implements the interface LoggerSObjectHandlerPlugin - -Flow: The API name of a Flow that supports these inputs: - - triggerOperationType: text variable - - triggerNew: record collection variable - - triggerOld: record collection variable - - 255 - true - Text - false - diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml b/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml deleted file mode 100644 index 2eef81563..000000000 --- a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/validationRules/Plugins_are_not_supported.validationRule-meta.xml +++ /dev/null @@ -1,14 +0,0 @@ - - - Plugins_are_not_supported - true - AND( - SObjectType__r.DeveloperName != 'LogEntryEvent', - SObjectType__r.DeveloperName != 'Log', - SObjectType__r.DeveloperName != 'LogEntry', - SObjectType__r.DeveloperName != 'LogEntryTag', - SObjectType__r.DeveloperName != 'LoggerTag' -) - SObjectType__c - Plugins are not supported for this SObject Handler - diff --git a/nebula-logger/core/tests/LoggerCore.testSuite-meta.xml b/nebula-logger/core/tests/LoggerCore.testSuite-meta.xml new file mode 100644 index 000000000..f1e0c16a2 --- /dev/null +++ b/nebula-logger/core/tests/LoggerCore.testSuite-meta.xml @@ -0,0 +1,30 @@ + + + ComponentLogger_Tests + FlowCollectionLogEntry_Tests + FlowLogEntry_Tests + FlowLogger_Tests + FlowRecordLogEntry_Tests + LogBatchPurger_Tests + LogBatchPurgeScheduler_Tests + LogEntryEventBuilder_Tests + LogEntryEventHandler_Tests + LogEntryFieldSetPicklist_Tests + LogEntryHandler_Tests + LogEntryTagHandler_Tests + LoggerBatchableContext_Tests + LoggerDataStore_Tests + LoggerEmailSender_Tests + LoggerParameter_Tests + LoggerPlugin_Tests + LoggerSettingsController_Tests + LoggerSObjectHandler_Tests + LoggerSObjectMetadata_Tests + LoggerTagHandler_Tests + LoggerTriggerableContext_Tests + Logger_Tests + LogHandler_Tests + LogMassDeleteExtension_Tests + LogMessage_Tests + RelatedLogEntriesController_Tests + diff --git a/nebula-logger/core/tests/common/classes/LoggerTestUtils.cls b/nebula-logger/core/tests/common/classes/LoggerTestUtils.cls deleted file mode 100644 index 7f172c947..000000000 --- a/nebula-logger/core/tests/common/classes/LoggerTestUtils.cls +++ /dev/null @@ -1,256 +0,0 @@ -//------------------------------------------------------------------------------------------------// -// This file is part of the Nebula Logger project, released under the MIT License. // -// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // -//------------------------------------------------------------------------------------------------// - -@SuppressWarnings('PMD.MethodNamingConventions, PMD.PropertyNamingConventions') -@IsTest -public class LoggerTestUtils { - private static final Map SOBJECT_TYPE_TO_MOCK_ID_COUNT = new Map(); - - // Lazy-load & cache permission sets & profiles - private static final Map PERMISSION_SETS_BY_NAME { - get { - if (PERMISSION_SETS_BY_NAME == null) { - PERMISSION_SETS_BY_NAME = queryPermissionSets(); - } - return PERMISSION_SETS_BY_NAME; - } - private set; - } - - private static final PermissionSet LOGGER_ADMIN_PERMISSION_SET { - get { - if (LOGGER_ADMIN_PERMISSION_SET == null) { - LOGGER_ADMIN_PERMISSION_SET = PERMISSION_SETS_BY_NAME.get('LoggerAdmin'); - } - return LOGGER_ADMIN_PERMISSION_SET; - } - private set; - } - - private static final PermissionSet LOGGER_LOG_VIEWER_PERMISSION_SET { - get { - if (LOGGER_LOG_VIEWER_PERMISSION_SET == null) { - LOGGER_LOG_VIEWER_PERMISSION_SET = PERMISSION_SETS_BY_NAME.get('LoggerLogViewer'); - } - return LOGGER_LOG_VIEWER_PERMISSION_SET; - } - private set; - } - - private static final PermissionSet LOGGER_END_USER_PERMISSION_SET { - get { - if (LOGGER_END_USER_PERMISSION_SET == null) { - LOGGER_END_USER_PERMISSION_SET = PERMISSION_SETS_BY_NAME.get('LoggerEndUser'); - } - return LOGGER_END_USER_PERMISSION_SET; - } - private set; - } - - private static final PermissionSet LOGGER_LOG_CREATOR_PERMISSION_SET { - get { - if (LOGGER_LOG_CREATOR_PERMISSION_SET == null) { - LOGGER_LOG_CREATOR_PERMISSION_SET = PERMISSION_SETS_BY_NAME.get('LoggerLogCreator'); - } - return LOGGER_LOG_CREATOR_PERMISSION_SET; - } - private set; - } - - /** - * @description Assigns the permission set `LoggerAdmin` to the specified user ID - * @param userId The ID of the user that should be assigned the permission set - */ - public static void assignAdminPermissionSet(Id userId) { - assignPermissionSet(userId, LOGGER_ADMIN_PERMISSION_SET); - } - - /** - * @description Assigns the permission set `LoggerLogViewer` to the specified user ID - * @param userId The ID of the user that should be assigned the permission set - */ - public static void assignLogViewerPermissionSet(Id userId) { - assignPermissionSet(userId, LOGGER_LOG_VIEWER_PERMISSION_SET); - } - - /** - * @description Assigns the permission set `LoggerEndUser` to the specified user ID - * @param userId The ID of the user that should be assigned the permission set - */ - public static void assignEndUserPermissionSet(Id userId) { - assignPermissionSet(userId, LOGGER_END_USER_PERMISSION_SET); - } - - /** - * @description Assigns the permission set `LoggerLogCreator` to the specified user ID - * @param userId The ID of the user that should be assigned the permission set - */ - public static void assignLogCreatorPermissionSet(Id userId) { - assignPermissionSet(userId, LOGGER_LOG_CREATOR_PERMISSION_SET); - } - - /** - * @description Instances of `AggregateResult` can not be created directly in Apex. - * This method uses a workaround to generate a mock. - * @return The mock instance of `AggregateResult` - */ - public static AggregateResult createMockAggregateResult() { - Map defaultMockAggregateKeyValues = new Map{ - 'fieldAvg' => 62.5, - 'fieldMax' => 100, - 'fieldMin' => 25, - 'fieldCount' => 4 - }; - return createMockAggregateResult(defaultMockAggregateKeyValues); - } - - /** - * @description Instances of `AggregateResult` can not be created directly in Apex. - * This method uses a workaround to generate a mock, using the provided map of aliases & aggregate values - * @param mockAggregateKeyValues A map of aliases & aggregate values to use when creating the mock `AggregateResult` - * @return The mock instance of `AggregateResult` - */ - public static AggregateResult createMockAggregateResult(Map mockAggregateKeyValues) { - return (AggregateResult) JSON.deserialize(JSON.serialize(mockAggregateKeyValues), AggregateResult.class); - } - - /** - * @description Generates a mock record ID for the provided SObject Type - * @param sobjectType The SObject Type for the generated mock record ID - * @return The mock record ID for the specified SObject Type - */ - public static String createMockId(Schema.SObjectType sobjectType) { - Integer recordIdNumber = 1; - if (SOBJECT_TYPE_TO_MOCK_ID_COUNT.containsKey(sobjectType)) { - recordIdNumber = SOBJECT_TYPE_TO_MOCK_ID_COUNT.get(sobjectType); - } - String recordIdSuffix = String.valueOf(recordIdNumber++); - SOBJECT_TYPE_TO_MOCK_ID_COUNT.put(sobjectType, recordIdNumber); - - String recordIdKeyPrefix = sobjectType.getDescribe().getKeyPrefix(); - Integer idFieldLength = sobjectType.getDescribe().fields.getMap().get('Id').getDescribe().getLength(); - Integer recordIdCenterLength = idFieldLength - recordIdKeyPrefix.length() - recordIdSuffix.length(); - return recordIdKeyPrefix + '0'.repeat(recordIdCenterLength) + recordIdSuffix; - } - - /** - * @description Creates a `User` record for testing purposes, using the current user's profile - * @return The generated `User` record - it is not automatically inserted into the database. - */ - public static User createUser() { - return createUser(UserInfo.getProfileId()); - } - - /** - * @description Creates a `User` record for testing purposes, using the specified profile ID - * @param profileId The `Profile` ID to use for the created `User` - * @return The generated `User` record - it is not automatically inserted into the database. - */ - public static User createUser(Id profileId) { - return new User( - Alias = 'log_xyz', - Email = 'logger_xyz@test.com.net.org', - EmailEncodingKey = 'ISO-8859-1', - LanguageLocaleKey = 'en_US', - LastName = 'Logger test user', - LocaleSidKey = 'en_US', - ProfileId = profileId, - TimeZoneSidKey = 'America/Los_Angeles', - Username = 'logger_xyz@test.com.net.org' - ); - } - - /** - * @description Queries for the `Organization` record for the current environment. - * @return The matching `Organization` record - */ - public static Organization getOrganization() { - return [SELECT Id, Name, InstanceName, IsSandbox, NamespacePrefix, OrganizationType, TrialExpirationDate FROM Organization]; - } - - /** - * @description Returns the current environment's type - Scratch Org, Sandbox, or Production. - * @return The environment type - */ - public static String getOrganizationEnvironmentType() { - Organization organization = getOrganization(); - - String orgEnvironmentType; - if (organization.IsSandbox == true && organization.TrialExpirationDate != null) { - orgEnvironmentType = 'Scratch Org'; - } else if (organization.IsSandbox == true) { - orgEnvironmentType = 'Sandbox'; - } else { - orgEnvironmentType = 'Production'; - } - return orgEnvironmentType; - } - - /** - * @description Returns the current user's `Network` (Experience Cloud site) - * @return The matching `Network` record - */ - public static SObject getNetwork() { - if (Network.getNetworkId() == null) { - return null; - } - - String networkApiName = 'Network'; - // Networks (communities) may not be enabled in the org (no Network object), so run everything dynamically - Boolean networksEnabled = Schema.getGlobalDescribe().containsKey(networkApiName); - - if (!networksEnabled) { - return null; - } - - String query = 'SELECT Id, Name, UrlPathPrefix FROM Network WHERE Id = :Network.getNetworkId()'; - return Database.query(String.escapeSingleQuotes(query)); - } - - /** - * @description Returns the current user - * @return The matching `User` record - */ - public static User getCurrentUser() { - return [ - SELECT Id, Profile.Name, Profile.UserLicenseId, Profile.UserLicense.LicenseDefinitionKey, Profile.UserLicense.Name, Username, UserRole.Name - FROM User - WHERE Id = :UserInfo.getUserId() - ]; - } - - /** - * @description Creates and inserts a `Group` record for testing queues, using the specified SObject Type - * @param sobjectType The SObjectType that the queue should be able to own (stored in `QueueSObject`) - * @return The inserted `Group` record - it is automatically inserted into the database, as well as 1 child `QueueSObject` record. - */ - public static Group insertQueue(Schema.SObjectType sobjectType) { - Group loggerQueue = new Group(DeveloperName = 'Some_Log_Queue', Name = 'Some Log Queue', Type = 'Queue'); - insert loggerQueue; - - // To avoid a MIXED_DML_OPERATION exception, use System.runs() for inserting the QueueSObject record - System.runAs(new User(Id = UserInfo.getUserId())) { - QueueSObject loggerQueueSObject = new QueueSObject(QueueId = loggerQueue.Id, SObjectType = sobjectType.getDescribe().getName()); - insert loggerQueueSObject; - } - - return loggerQueue; - } - - // Helper methods - private static void assignPermissionSet(Id userId, PermissionSet permissionSet) { - PermissionSetAssignment permissionSetAssignment = new PermissionSetAssignment(AssigneeId = userId, PermissionSetId = permissionSet.Id); - insert permissionSetAssignment; - } - - private static Map queryPermissionSets() { - List permissionSetNames = new List{ 'LoggerAdmin', 'LoggerLogViewer', 'LoggerEndUser', 'LoggerLogCreator' }; - Map results = new Map(); - for (PermissionSet permissionSet : [SELECT Id, Name FROM PermissionSet WHERE Name IN :permissionSetNames]) { - results.put(permissionSet.Name, permissionSet); - } - return results; - } -} diff --git a/nebula-logger/core/tests/configuration/classes/LoggerEmailUtils_Tests.cls b/nebula-logger/core/tests/configuration/classes/LoggerEmailUtils_Tests.cls deleted file mode 100644 index 1c9250291..000000000 --- a/nebula-logger/core/tests/configuration/classes/LoggerEmailUtils_Tests.cls +++ /dev/null @@ -1,102 +0,0 @@ -//------------------------------------------------------------------------------------------------// -// This file is part of the Nebula Logger project, released under the MIT License. // -// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // -//------------------------------------------------------------------------------------------------// - -@SuppressWarnings('PMD.MethodNamingConventions') -@IsTest -private class LoggerEmailUtils_Tests { - @IsTest - static void it_should_send_email_notification_for_saveResult_errors_when_enabled() { - LoggerEmailUtils.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); - - // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (interntionally) fail - List saveResultsWithErrors = Database.insert(new List{ new LogEntry__c() }, false); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResultsWithErrors); - - System.assertEquals(1, Limits.getEmailInvocations(), 'Email should have been sent'); - System.assertEquals( - true, - LoggerEmailUtils.SENT_EMAILS.get(0).getHtmlBody().contains(saveResultsWithErrors.get(0).errors.get(0).getMessage()), - 'Email message should contain SaveResult error message' - ); - } - - @IsTest - static void it_should_not_send_email_notification_for_saveResult_errors_when_no_recipients_configured() { - LoggerParameter.setMockParameter(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'true')); - System.assertEquals(true, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); - LoggerEmailUtils.CACHED_APEX_ERROR_RECIPIENTS.clear(); - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); - - // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (interntionally) fail - List saveResultsWithErrors = Database.insert(new List{ new LogEntry__c() }, false); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResultsWithErrors); - - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); - System.assertEquals(true, LoggerEmailUtils.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); - } - - @IsTest - static void it_should_not_send_email_notification_for_saveResult_errors_when_disabled() { - LoggerParameter.setMockParameter(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'false')); - System.assertEquals(false, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); - LoggerEmailUtils.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); - - // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (interntionally) fail - List saveResultsWithErrors = Database.insert(new List{ new LogEntry__c() }, false); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResultsWithErrors); - - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); - System.assertEquals(true, LoggerEmailUtils.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); - } - - @IsTest - static void it_should_send_email_notification_for_upsertResult_errors_when_enabled() { - LoggerEmailUtils.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); - - // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (interntionally) fail - List upsertResultsWithErrors = Database.upsert(new List{ new LogEntry__c() }, false); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntry__c.SObjectType, upsertResultsWithErrors); - - System.assertEquals(1, Limits.getEmailInvocations(), 'Email should have been sent'); - System.assertEquals( - true, - LoggerEmailUtils.SENT_EMAILS.get(0).getHtmlBody().contains(upsertResultsWithErrors.get(0).errors.get(0).getMessage()), - 'Email message should contain UpsertResult error message' - ); - } - - @IsTest - static void it_should_not_send_email_notification_for_upsertResult_errors_when_no_recipients_configured() { - LoggerParameter.setMockParameter(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'true')); - System.assertEquals(true, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); - LoggerEmailUtils.CACHED_APEX_ERROR_RECIPIENTS.clear(); - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); - - // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (interntionally) fail - List upsertResultsWithErrors = Database.upsert(new List{ new LogEntry__c() }, false); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntry__c.SObjectType, upsertResultsWithErrors); - - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); - System.assertEquals(true, LoggerEmailUtils.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); - } - - @IsTest - static void it_should_not_send_email_notification_for_upsertResult_errors_when_disabled() { - LoggerParameter.setMockParameter(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'false')); - System.assertEquals(false, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); - LoggerEmailUtils.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); - - // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (interntionally) fail - List upsertResultsWithErrors = Database.upsert(new List{ new LogEntry__c() }, false); - LoggerEmailUtils.sendErrorEmail(Schema.LogEntry__c.SObjectType, upsertResultsWithErrors); - - System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); - System.assertEquals(true, LoggerEmailUtils.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); - } -} diff --git a/nebula-logger/core/tests/configuration/classes/LoggerParameter_Tests.cls b/nebula-logger/core/tests/configuration/classes/LoggerParameter_Tests.cls index 447915dc3..265c32cf8 100644 --- a/nebula-logger/core/tests/configuration/classes/LoggerParameter_Tests.cls +++ b/nebula-logger/core/tests/configuration/classes/LoggerParameter_Tests.cls @@ -4,19 +4,112 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LoggerParameter_Tests { private static User getUserRecord() { return new User(Id = UserInfo.getUserId(), Username = UserInfo.getUserName()); } + @IsTest + static void it_should_throw_exception_when_mock_record_developer_name_is_null() { + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = null); + Exception thrownIllegalArgumentException; + + try { + LoggerParameter.setMock(mockParameter); + } catch (IllegalArgumentException ex) { + thrownIllegalArgumentException = ex; + } + + System.assertNotEquals(null, thrownIllegalArgumentException); + } + + @IsTest + static void it_should_load_mock_record() { + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MockParameter', Value__c = 'Hello, world!'); + LoggerParameter.setMock(mockParameter); + + String returnedValue = LoggerParameter.getString(mockParameter.DeveloperName, null); + + System.assertNotEquals(null, returnedValue); + System.assertEquals(mockParameter.Value__c, returnedValue); + } + + @IsTest + static void it_should_return_constant_value_for_call_status_api() { + Boolean mockValue = true; + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'CallStatusApi', Value__c = JSON.serialize(mockValue)); + LoggerParameter.setMock(mockParameter); + + Boolean returnedValue = LoggerParameter.CALL_STATUS_API; + + System.assertEquals(mockValue, returnedValue); + } + + @IsTest + static void it_should_return_constant_value_for_enable_system_messages() { + Boolean mockValue = true; + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = JSON.serialize(mockValue)); + LoggerParameter.setMock(mockParameter); + + Boolean returnedValue = LoggerParameter.ENABLE_SYSTEM_MESSAGES; + + System.assertEquals(mockValue, returnedValue); + } + + @IsTest + static void it_should_return_constant_value_for_send_error_email_notifications() { + Boolean mockValue = false; + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = JSON.serialize(mockValue)); + LoggerParameter.setMock(mockParameter); + + Boolean returnedValue = LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS; + + System.assertEquals(mockValue, returnedValue); + } + + @IsTest + static void it_should_return_constant_value_for_system_debug_message_format() { + // The constant LoggerParameter.SYSTEM_DEBUG_MESSAGE_FORMAT is unique - its value should always be + // loaded from LoggerParameter__mdt, even in a test context so that orgs can control the output of debug + // messages within their own test classes. All other constants use mock configurations + String configuredValue = LoggerParameter__mdt.getInstance('SystemDebugMessageFormat').Value__c; + + String returnedValue = LoggerParameter.SYSTEM_DEBUG_MESSAGE_FORMAT; + + System.assertEquals(configuredValue, returnedValue); + } + + @IsTest + static void it_should_return_constant_value_for_tagging_is_enabled() { + Boolean mockValue = false; + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'EnableTagging', Value__c = JSON.serialize(mockValue)); + LoggerParameter.setMock(mockParameter); + + Boolean returnedValue = LoggerParameter.TAGGING_IS_ENABLED; + + System.assertEquals(mockValue, returnedValue); + } + + @IsTest + static void it_should_return_constant_value_for_tag_using_topics() { + Boolean mockValue = true; + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'UseTopicsForTags', Value__c = JSON.serialize(mockValue)); + LoggerParameter.setMock(mockParameter); + + Boolean returnedValue = LoggerParameter.TAG_USING_TOPICS; + + System.assertEquals(mockValue, returnedValue); + } + @IsTest static void it_should_return_boolean_parameter() { Boolean parameterValue = true; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyBoolean', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Boolean returnedValue = LoggerParameter.getBoolean(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -24,19 +117,35 @@ private class LoggerParameter_Tests { static void it_should_return_boolean_list_parameter() { List parameterValue = new List{ true, false, true, true }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyBooleanList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getBooleanList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } + @IsTest + static void it_should_return_default_value_when_configured_boolean_list_parameter_is_null() { + List parameterValue = null; + LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyBooleanList', Value__c = JSON.serialize(parameterValue)); + LoggerParameter.setMock(mockParameter); + List defaultParameterValue = new List{ true, false }; + + List returnedValue = LoggerParameter.getBooleanList(mockParameter.DeveloperName, defaultParameterValue); + + System.assertEquals(null, parameterValue); + System.assertNotEquals(parameterValue, returnedValue, 'Returned value should not match configured null parameter value'); + System.assertEquals(defaultParameterValue, returnedValue, 'Returned value does not match expected default parameter value'); + } + @IsTest static void it_should_return_date_parameter() { Date parameterValue = System.today().addDays(-7); LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDate', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Date returnedValue = LoggerParameter.getDate(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -44,9 +153,10 @@ private class LoggerParameter_Tests { static void it_should_return_date_list_parameter() { List parameterValue = new List{ System.today(), System.today().addDays(10) }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDateList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getDateList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -54,9 +164,10 @@ private class LoggerParameter_Tests { static void it_should_return_datetime_parameter() { Datetime parameterValue = System.now().addDays(-7); LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDatetime', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Datetime returnedValue = LoggerParameter.getDatetime(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -64,9 +175,10 @@ private class LoggerParameter_Tests { static void it_should_return_datetime_list_parameter() { List parameterValue = new List{ System.now(), System.now().addDays(10) }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDatetimeList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getDatetimeList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -74,9 +186,10 @@ private class LoggerParameter_Tests { static void it_should_return_decimal_parameter() { Decimal parameterValue = 123456.0987; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDecimal', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Decimal returnedValue = LoggerParameter.getDecimal(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -84,9 +197,10 @@ private class LoggerParameter_Tests { static void it_should_return_decimal_list_parameter() { List parameterValue = new List{ 123.45, 678.09 }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDecimalList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getDecimalList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -94,9 +208,10 @@ private class LoggerParameter_Tests { static void it_should_return_double_parameter() { Double parameterValue = 123456.0987; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDouble', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Double returnedValue = LoggerParameter.getDouble(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -104,9 +219,10 @@ private class LoggerParameter_Tests { static void it_should_return_double_list_parameter() { List parameterValue = new List{ 123.45, 678.09 }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyDoubleList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getDoubleList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -114,9 +230,10 @@ private class LoggerParameter_Tests { static void it_should_return_id_parameter() { Id parameterValue = UserInfo.getUserId(); LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyId', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Id returnedValue = LoggerParameter.getId(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -124,9 +241,10 @@ private class LoggerParameter_Tests { static void it_should_return_id_list_parameter() { List parameterValue = new List{ UserInfo.getUserId() }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyIdList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getIdList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -134,9 +252,10 @@ private class LoggerParameter_Tests { static void it_should_return_integer_parameter() { Integer parameterValue = 123456; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyInteger', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Integer returnedValue = LoggerParameter.getInteger(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -144,9 +263,10 @@ private class LoggerParameter_Tests { static void it_should_return_integer_list_parameter() { List parameterValue = new List{ 12345, 67809 }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyIntegerList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getIntegerList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -154,9 +274,10 @@ private class LoggerParameter_Tests { static void it_should_return_long_parameter() { Long parameterValue = 123456; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyLong', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); Long returnedValue = LoggerParameter.getLong(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -164,9 +285,10 @@ private class LoggerParameter_Tests { static void it_should_return_long_list_parameter() { List parameterValue = new List{ 12345, 67809 }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyLongList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getLongList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -174,9 +296,10 @@ private class LoggerParameter_Tests { static void it_should_return_sobject_parameter() { SObject parameterValue = getUserRecord(); LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MySObject', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); SObject returnedValue = LoggerParameter.getSObject(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -184,9 +307,10 @@ private class LoggerParameter_Tests { static void it_should_return_sobject_list_parameter() { List parameterValue = new List{ getUserRecord() }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MySObjectList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getSObjectList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -194,9 +318,10 @@ private class LoggerParameter_Tests { static void it_should_return_string_parameter() { String parameterValue = 'Hello'; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyString', Value__c = parameterValue); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); String returnedValue = LoggerParameter.getString(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } @@ -204,9 +329,33 @@ private class LoggerParameter_Tests { static void it_should_return_string_list_parameter() { List parameterValue = new List{ 'Hello', 'Goodbye' }; LoggerParameter__mdt mockParameter = new LoggerParameter__mdt(DeveloperName = 'MyStringList', Value__c = JSON.serialize(parameterValue)); - LoggerParameter.setMockParameter(mockParameter); + LoggerParameter.setMock(mockParameter); List returnedValue = LoggerParameter.getStringList(mockParameter.DeveloperName, null); + System.assertEquals(parameterValue, returnedValue, 'Returned value does not match expected parameter value'); } + + @IsTest + static void it_should_match_on_developer_name_prefix() { + String developerNamePrefix = 'SomePrefix'; + LoggerParameter__mdt firstMatchingMockParameter = new LoggerParameter__mdt(DeveloperName = developerNamePrefix + 'SomeExample', Value__c = 'hello'); + LoggerParameter.setMock(firstMatchingMockParameter); + LoggerParameter__mdt secondMatchingMockParameter = new LoggerParameter__mdt(DeveloperName = developerNamePrefix + 'AnotherExample', Value__c = 'world'); + LoggerParameter.setMock(secondMatchingMockParameter); + LoggerParameter__mdt nonMatchingMockParameter = new LoggerParameter__mdt(DeveloperName = 'ThisShouldNotMatch', Value__c = '!'); + LoggerParameter.setMock(nonMatchingMockParameter); + + List matchingParameters = LoggerParameter.matchOnPrefix(developerNamePrefix); + + System.assertEquals(2, matchingParameters.size()); + for (LoggerParameter__mdt matchingParameter : matchingParameters) { + System.assertEquals( + true, + matchingParameter.DeveloperName == firstMatchingMockParameter.DeveloperName || + matchingParameter.DeveloperName == secondMatchingMockParameter.DeveloperName, + 'Returned parameter should be one of the two matching mock parameters' + ); + } + } } diff --git a/nebula-logger/core/tests/configuration/classes/LoggerPlugin_Tests.cls b/nebula-logger/core/tests/configuration/classes/LoggerPlugin_Tests.cls new file mode 100644 index 000000000..a0e95dc76 --- /dev/null +++ b/nebula-logger/core/tests/configuration/classes/LoggerPlugin_Tests.cls @@ -0,0 +1,123 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') +@IsTest(IsParallel=true) +private class LoggerPlugin_Tests { + private static final String PLUGIN_LOG_STATUS = 'On Hold'; + + @IsTest + static void it_should_returned_filtered_plugin_configurations_in_sorted_order() { + // The mock LoggerPlugin__mdt records are purposeful being added to LoggerPlugin.setMock() + // in an order that's different from the order when sorting on SObjectHandlerExecutionOrder__c. + // This helps ensure that the returned results are truly being sorted, instead of just being returned + // in the same order that they're added via LoggerPlugin.setMock() + LoggerPlugin__mdt secondPluginConfiguration = createMockPluginConfiguration('secondPluginConfiguration'); + secondPluginConfiguration.SObjectHandlerApexClass__c = 'Also_A_Fake_Apex_Class'; + secondPluginConfiguration.SObjectHandlerExecutionOrder__c = 2; + LoggerPlugin.setMock(secondPluginConfiguration); + LoggerPlugin__mdt fifthPluginConfiguration = createMockPluginConfiguration('zzzzz_sort_me_last_alphabetically_fifthPluginConfiguration'); + fifthPluginConfiguration.SObjectHandlerApexClass__c = 'Yeah_This_Is_A_Fake_Apex_Class'; + fifthPluginConfiguration.SObjectHandlerExecutionOrder__c = null; + LoggerPlugin.setMock(fifthPluginConfiguration); + LoggerPlugin__mdt fourthPluginConfiguration = createMockPluginConfiguration('fourthPluginConfiguration'); + fourthPluginConfiguration.SObjectHandlerApexClass__c = 'Yeah_This_Is_A_Fake_Apex_Class'; + fourthPluginConfiguration.SObjectHandlerExecutionOrder__c = null; + LoggerPlugin.setMock(fourthPluginConfiguration); + LoggerPlugin__mdt thirdPluginConfiguration = createMockPluginConfiguration('thirdPluginConfiguration'); + thirdPluginConfiguration.SObjectHandlerApexClass__c = 'Yeah_This_Is_A_Fake_Apex_Class'; + thirdPluginConfiguration.SObjectHandlerExecutionOrder__c = 3; + LoggerPlugin.setMock(thirdPluginConfiguration); + LoggerPlugin__mdt firstPluginConfiguration = createMockPluginConfiguration('firstPluginConfiguration'); + firstPluginConfiguration.SObjectHandlerApexClass__c = 'Some_Fake_Apex_Class'; + firstPluginConfiguration.SObjectHandlerExecutionOrder__c = 1; + LoggerPlugin.setMock(firstPluginConfiguration); + List expectedSortedPluginConfigurations = new List{ + firstPluginConfiguration, + secondPluginConfiguration, + thirdPluginConfiguration, + fourthPluginConfiguration, + fifthPluginConfiguration + }; + LoggerPlugin__mdt disabledPluginConfiguration = createMockPluginConfiguration('disabledPluginConfiguration'); + secondPluginConfiguration.IsEnabled__c = false; + secondPluginConfiguration.SObjectHandlerApexClass__c = 'Also_A_Fake_Apex_Class'; + secondPluginConfiguration.SObjectHandlerExecutionOrder__c = 2; + LoggerPlugin.setMock(disabledPluginConfiguration); + LoggerPlugin__mdt nonMatchingPluginConfiguration = createMockPluginConfiguration('nonMatchingPluginConfiguration'); + nonMatchingPluginConfiguration.SObjectHandlerApexClass__c = null; + nonMatchingPluginConfiguration.SObjectHandlerExecutionOrder__c = 1; + LoggerPlugin.setMock(nonMatchingPluginConfiguration); + + List returnedPluginConfigurations = LoggerPlugin.getFilteredPluginConfigurations( + new List{ Schema.LoggerPlugin__mdt.SObjectHandlerApexClass__c }, + Schema.LoggerPlugin__mdt.SObjectHandlerExecutionOrder__c + ); + + System.assertEquals(expectedSortedPluginConfigurations.size(), returnedPluginConfigurations.size()); + for (Integer i = 0; i < returnedPluginConfigurations.size(); i++) { + LoggerPlugin__mdt expectedSortedPluginConfiguration = expectedSortedPluginConfigurations.get(i); + LoggerPlugin__mdt returnedPluginConfiguration = returnedPluginConfigurations.get(i); + System.assertEquals(expectedSortedPluginConfiguration, returnedPluginConfiguration, 'Records at index ' + i + ' don\'t match'); + } + } + + @IsTest + static void it_should_return_batchable_apex_plugin_instance_for_valid_class() { + LoggerPlugin.Batchable batchableApexPlugin = LoggerPlugin.newBatchableInstance(ExampleBatchPurgerPlugin.class.getName()); + + System.assertNotEquals(null, batchableApexPlugin, ExampleBatchPurgerPlugin.class.getName()); + System.assertEquals(true, batchableApexPlugin instanceof ExampleBatchPurgerPlugin); + } + + @IsTest + static void it_should_return_null_batchable_apex_plugin_instance_for_invalid_class() { + LoggerPlugin.Batchable batchableApexPlugin = LoggerPlugin.newBatchableInstance('Some fake class, this definitely doesn\'t exist'); + + System.assertEquals(null, batchableApexPlugin); + } + + @IsTest + static void it_should_return_triggerable_apex_plugin_instance_for_valid_class() { + LoggerPlugin.Triggerable triggerableApexPlugin = LoggerPlugin.newTriggerableInstance(ExampleTriggerablePlugin.class.getName()); + + System.assertNotEquals(null, triggerableApexPlugin, ExampleTriggerablePlugin.class.getName()); + System.assertEquals(true, triggerableApexPlugin instanceof ExampleTriggerablePlugin); + } + + @IsTest + static void it_should_return_null_triggerable_apex_plugin_instance_for_invalid_class() { + LoggerPlugin.Triggerable triggerableApexPlugin = LoggerPlugin.newTriggerableInstance('Some fake class, this definitely doesn\'t exist'); + + System.assertEquals(null, triggerableApexPlugin); + } + + private static LoggerPlugin__mdt createMockPluginConfiguration(String developerName) { + return new LoggerPlugin__mdt(DeveloperName = developerName, IsEnabled__c = true); + } + + @SuppressWarnings('PMD.ApexDoc, PMD.EmptyStatementBlock') + public class ExampleBatchPurgerPlugin implements LoggerPlugin.Batchable { + public void start(LoggerPlugin__mdt configuration, LoggerBatchableContext input) { + } + public void execute(LoggerPlugin__mdt configuration, LoggerBatchableContext input, List scope) { + } + public void finish(LoggerPlugin__mdt configuration, LoggerBatchableContext input) { + } + } + + @SuppressWarnings('PMD.ApexDoc, PMD.EmptyStatementBlock') + public class ExampleTriggerablePlugin implements LoggerPlugin.Triggerable { + public void execute(LoggerPlugin__mdt configuration, LoggerTriggerableContext input) { + switch on input.triggerOperationType { + when BEFORE_INSERT { + for (Log__c log : (List) input.triggerNew) { + log.Status__c = PLUGIN_LOG_STATUS; + } + } + } + } + } +} diff --git a/nebula-logger/extra-tests/tests/LogHandler_Tests_Flow.cls-meta.xml b/nebula-logger/core/tests/configuration/classes/LoggerPlugin_Tests.cls-meta.xml similarity index 100% rename from nebula-logger/extra-tests/tests/LogHandler_Tests_Flow.cls-meta.xml rename to nebula-logger/core/tests/configuration/classes/LoggerPlugin_Tests.cls-meta.xml diff --git a/nebula-logger/core/tests/configuration/testSuites/LoggerConfiguration.testSuite-meta.xml b/nebula-logger/core/tests/configuration/testSuites/LoggerConfiguration.testSuite-meta.xml index cf5d52783..17a87cb0c 100644 --- a/nebula-logger/core/tests/configuration/testSuites/LoggerConfiguration.testSuite-meta.xml +++ b/nebula-logger/core/tests/configuration/testSuites/LoggerConfiguration.testSuite-meta.xml @@ -1,4 +1,5 @@ LoggerParameter_Tests + LoggerPlugin_Tests diff --git a/nebula-logger/core/tests/configuration/utilities/LoggerMockDataCreator.cls b/nebula-logger/core/tests/configuration/utilities/LoggerMockDataCreator.cls new file mode 100644 index 000000000..a1a6d5de3 --- /dev/null +++ b/nebula-logger/core/tests/configuration/utilities/LoggerMockDataCreator.cls @@ -0,0 +1,678 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Test Utilities + * @description Utility class used to help with generating mock data when writing Apex tests for Nebula Logger. + * These methods are generic, and should work in any Salesforce org. + * These methods can be used when writing Apex tests for plugins. + * @see LoggerMockDataStore + * @see LoggerTestConfigurator + */ +@SuppressWarnings('PMD.CyclomaticComplexity, PMD.ExcessivePublicCount, PMD.MethodNamingConventions, PMD.PropertyNamingConventions') +@IsTest +public class LoggerMockDataCreator { + private static final Map> SOBJECT_TYPE_TO_ALL_FIELDS = new Map>(); + private static final Map> SOBJECT_TYPE_TO_REQUIRED_FIELDS = new Map>(); + private static final Map SOBJECT_TYPE_TO_MOCK_ID_COUNT = new Map(); + + private static Integer userMockUsernameCount = 0; + + /** + * @description Instances of `AggregateResult` can not be created directly in Apex. + * This method uses a workaround to generate a mock. + * @return The mock instance of `AggregateResult` + */ + public static AggregateResult createAggregateResult() { + Map defaultMockAggregateKeyValues = new Map{ + 'fieldAvg' => 62.5, + 'fieldMax' => 100, + 'fieldMin' => 25, + 'fieldCount' => 4 + }; + return createAggregateResult(defaultMockAggregateKeyValues); + } + + /** + * @description Instances of `AggregateResult` can not be created directly in Apex. + * This method uses a workaround to generate a mock, using the provided map of aliases & aggregate values + * @param mockAggregateKeyValues A map of aliases & aggregate values to use when creating the mock `AggregateResult` + * @return The mock instance of `AggregateResult` + */ + public static AggregateResult createAggregateResult(Map mockAggregateKeyValues) { + return (AggregateResult) JSON.deserialize(JSON.serialize(mockAggregateKeyValues), AggregateResult.class); + } + + /** + * @description Creates an instance of the class `MockBatchableContext` that implements the interface `Database.BatchableContext`. + * This can be used when testing batch jobs. + * @param jobId A string value to use as the batchable job ID - this can be a true ID, or just a string + * @return The instance of `MockBatchableContext` + */ + public static MockBatchableContext createBatchableContext(String jobId) { + return new MockBatchableContext(jobId); + } + + /** + * @description Creates a mock instance of `Database.DeleteResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests). A fake + * record ID is automatically included. + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @return The mock instance of `Database.DeleteResult` + */ + public static Database.DeleteResult createDatabaseDeleteResult(Boolean isSuccess) { + return createDatabaseDeleteResult(isSuccess, createId(Schema.Account.SObjectType)); + } + + /** + * @description Creates a mock instance of `Database.DeleteResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests) + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @param recordId The record ID to use within the mock result + * @return The mock instance of `Database.DeleteResult` + */ + public static Database.DeleteResult createDatabaseDeleteResult(Boolean isSuccess, Id recordId) { + if (isSuccess == true) { + return (Database.DeleteResult) JSON.deserialize('{"success": true, "id": "' + recordId + '"}', Database.DeleteResult.class); + } else { + return (Database.DeleteResult) JSON.deserialize( + '{"success":false,"errors":[{"message": "Could not delete...", "statusCode": "DELETE_FAILED"}]}', + Database.DeleteResult.class + ); + } + } + + /** + * @description Creates a mock instance of `Database.MergeResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests). A fake + * record ID is automatically included. + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @return The mock instance of `Database.MergeResult` + */ + public static Database.MergeResult createDatabaseMergeResult(Boolean isSuccess) { + return createDatabaseMergeResult(isSuccess, createId(Schema.Account.SObjectType)); + } + + /** + * @description Creates a mock instance of `Database.MergeResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests) + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @param recordId The record ID to use within the mock result + * @return The mock instance of `Database.MergeResult` + */ + public static Database.MergeResult createDatabaseMergeResult(Boolean isSuccess, Id recordId) { + if (isSuccess == true) { + return (Database.MergeResult) JSON.deserialize('{"success": true, "id": "' + recordId + '"}', Database.MergeResult.class); + } else { + return (Database.MergeResult) JSON.deserialize( + '{"success":false,"errors":[{"message": "Could not merge...", "statusCode": "MERGE_FAILED"}]}', + Database.MergeResult.class + ); + } + } + + /** + * @description Creates a mock instance of `Database.SaveResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests). A fake + * record ID is automatically included. + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @return The mock instance of `Database.SaveResult` + */ + public static Database.SaveResult createDatabaseSaveResult(Boolean isSuccess) { + return createDatabaseSaveResult(isSuccess, createId(Schema.Account.SObjectType)); + } + + /** + * @description Creates a mock instance of `Database.SaveResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests) + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @param recordId The record ID to use within the mock result + * @return The mock instance of `Database.SaveResult` + */ + public static Database.SaveResult createDatabaseSaveResult(Boolean isSuccess, Id recordId) { + if (isSuccess == true) { + return (Database.SaveResult) JSON.deserialize('{"success": true, "id": "' + recordId + '"}', Database.SaveResult.class); + } else { + return (Database.SaveResult) JSON.deserialize( + '{"success":false,"errors":[{"message": "Could not save...", "statusCode": "FIELD_CUSTOM_VALIDATION_EXCEPTION"}]}', + Database.SaveResult.class + ); + } + } + + /** + * @description Creates a mock instance of `Database.UndeleteResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests). A fake + * record ID is automatically included. + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @return The mock instance of `Database.UndeleteResult` + */ + public static Database.UndeleteResult createDatabaseUndeleteResult(Boolean isSuccess) { + return createDatabaseUndeleteResult(isSuccess, createId(Schema.Account.SObjectType)); + } + + /** + * @description Creates a mock instance of `Database.UndeleteResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests) + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @param recordId The record ID to use within the mock result + * @return The mock instance of `Database.UndeleteResult` + */ + public static Database.UndeleteResult createDatabaseUndeleteResult(Boolean isSuccess, Id recordId) { + if (isSuccess == true) { + return (Database.UndeleteResult) JSON.deserialize('{"success": true, "id": "' + recordId + '"}', Database.UndeleteResult.class); + } else { + return (Database.UndeleteResult) JSON.deserialize( + '{"success":false,"errors":[{"message": "Could not undelete...", "statusCode": "FIELD_CUSTOM_VALIDATION_EXCEPTION"}]}', + Database.UndeleteResult.class + ); + } + } + + /** + * @description Creates a mock instance of `Database.UpsertResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests). A fake + * record ID is automatically included. + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @param isCreated Indicates if the generated mock should have `isCreated` == true + * @return The mock instance of `Database.UpsertResult` + */ + public static Database.UpsertResult createDatabaseUpsertResult(Boolean isSuccess, Boolean isCreated) { + return createDatabaseUpsertResult(isSuccess, isCreated, createId(Schema.Account.SObjectType)); + } + + /** + * @description Creates a mock instance of `Database.UpsertResult` - a mock is used instead of an actual instance + * to help speed up tests, and to support writing unit tests (instead of integration tests) + * @param isSuccess Indicates if the generated mock should have `isSuccess` == true + * @param isCreated Indicates if the generated mock should have `isCreated` == true + * @param recordId The record ID to use within the mock result + * @return The mock instance of `Database.UpsertResult` + */ + public static Database.UpsertResult createDatabaseUpsertResult(Boolean isSuccess, Boolean isCreated, Id recordId) { + if (isSuccess == true) { + return (Database.UpsertResult) JSON.deserialize( + '{"success": ' + + isSuccess + + ', "created": ' + + isCreated + + ', "id": "' + + recordId + + '"}', + Database.UpsertResult.class + ); + } else { + return (Database.UpsertResult) JSON.deserialize( + '{"success":false, "created":' + + isCreated + + ', "errors":[{"message": "Could not upsert...", "statusCode": "FIELD_CUSTOM_VALIDATION_EXCEPTION"}]}', + Database.UpsertResult.class + ); + } + } + + /** + * @description Generates an instance of the class `MockHttpCallout` that implements the interface `HttpCalloutMock`. + * This can be used when testing batch jobs. + * @return The instance of `MockHttpCallout` + */ + public static MockHttpCallout createHttpCallout() { + return new MockHttpCallout(); + } + + /** + * @description Generates an instance of `HttpRequest`. This can be used when testing logging capabilities for instances of `HttpRequest`. + * @return The instance of `HttpRequest` + */ + public static HttpRequest createHttpRequest() { + HttpRequest request = new HttpRequest(); + request.setBody('Hello, world!'); + request.setCompressed(true); + request.setEndpoint('https://fake.salesforce.com'); + request.setMethod('GET'); + return request; + } + + /** + * @description Generates an instance of `HttpResponse`. This can be used when testing logging capabilities for instances of `HttpResponse`. + * @return The instance of `HttpResponse` + */ + public static HttpResponse createHttpResponse() { + HttpResponse response = new HttpResponse(); + response.setBody('Hello, world!'); + response.setHeader('someKey', 'some string value'); + response.setHeader('anotherKey', 'an amazing example value, wow'); + response.setStatus('STATUS_GOOD_JOB_YOU_DID_IT'); + response.setStatusCode(201); + return response; + } + + /** + * @description Generates a mock record ID for the provided SObject Type + * @param sobjectType The SObject Type for the generated mock record ID + * @return The mock record ID for the specified SObject Type + */ + public static String createId(Schema.SObjectType sobjectType) { + Integer recordIdNumber = 1; + if (SOBJECT_TYPE_TO_MOCK_ID_COUNT.containsKey(sobjectType)) { + recordIdNumber = SOBJECT_TYPE_TO_MOCK_ID_COUNT.get(sobjectType); + } + String recordIdSuffix = String.valueOf(recordIdNumber++); + SOBJECT_TYPE_TO_MOCK_ID_COUNT.put(sobjectType, recordIdNumber); + + String recordIdKeyPrefix = sobjectType.getDescribe().getKeyPrefix(); + Integer idFieldLength = sobjectType.getDescribe().fields.getMap().get('Id').getDescribe().getLength(); + Integer recordIdCenterLength = idFieldLength - recordIdKeyPrefix.length() - recordIdSuffix.length(); + return recordIdKeyPrefix + '0'.repeat(recordIdCenterLength) + recordIdSuffix; + } + + /** + * @description Creates a new builder instance for the specified `SObjectType`, including creating a + * new `SObject` record. The new `SObject` record is created with any default field values that + * have been configured on the `SObjectType`. + * @param sobjectType The `SObjectType` to use for generating a new test `SObject` record + * @return A new instance of `SObjectTestDataBuilder` for the specified `SObjectType` + */ + public static SObjectTestDataBuilder createDataBuilder(Schema.SObjectType sobjectType) { + return new SObjectTestDataBuilder(sobjectType); + } + + /** + * @description Creates a new builder instance for the specified `SObject` record + * @param record The existing test `SObject` record to populate with sample data + * @return A new instance of `SObjectTestDataBuilder` for the specified `SObject` + */ + public static SObjectTestDataBuilder createDataBuilder(SObject record) { + return new SObjectTestDataBuilder(record); + } + + /** + * @description Creates a `User` record for testing purposes, using the current user's profile + * @return The generated `User` record - it is not automatically inserted into the database. + */ + public static User createUser() { + return createUser(UserInfo.getProfileId()); + } + + /** + * @description Creates a `User` record for testing purposes, using the specified profile ID + * @param profileId The `Profile` ID to use for the created `User` + * @return The generated `User` record - it is not automatically inserted into the database. + */ + public static User createUser(Id profileId) { + return new User( + Alias = 'user_xyz', + Email = 'user_xyz@test.com.net.org', + EmailEncodingKey = 'ISO-8859-1', + LanguageLocaleKey = 'en_US', + LastName = 'Test Userson', + LocaleSidKey = 'en_US', + ProfileId = profileId, + TimeZoneSidKey = 'America/Los_Angeles', + Username = 'user_xyz' + (userMockUsernameCount++) + '@test.com.net.org' + ); + } + + /** + * @description Queries for the `Organization` record for the current environment. + * @return The matching `Organization` record + */ + public static Organization getOrganization() { + return [SELECT Id, Name, InstanceName, IsSandbox, NamespacePrefix, OrganizationType, TrialExpirationDate FROM Organization]; + } + + /** + * @description Returns the current environment's type - Scratch Org, Sandbox, or Production. + * @return The environment type + */ + public static String getOrganizationEnvironmentType() { + Organization organization = getOrganization(); + + String orgEnvironmentType; + if (organization.IsSandbox == true && organization.TrialExpirationDate != null) { + orgEnvironmentType = 'Scratch Org'; + } else if (organization.IsSandbox == true) { + orgEnvironmentType = 'Sandbox'; + } else { + orgEnvironmentType = 'Production'; + } + return orgEnvironmentType; + } + + /** + * @description Returns the current user's `Network` (Experience Cloud site) + * @return The matching `Network` record + */ + public static SObject getNetwork() { + if (Network.getNetworkId() == null || Type.forName('Network') == null) { + return null; + } + + String query = 'SELECT Id, Name, UrlPathPrefix FROM Network WHERE Id = :Network.getNetworkId()'; + return Database.query(String.escapeSingleQuotes(query)); + } + + /** + * @description Returns the current user + * @return The matching `User` record + */ + public static User getUser() { + return getUser(UserInfo.getUserId()); + } + + /** + * @description Returns the specified user + * @param userId The ID of the `User` record to query + * @return The matching `User` record + */ + public static User getUser(Id userId) { + return [ + SELECT Id, Profile.Name, Profile.UserLicenseId, Profile.UserLicense.LicenseDefinitionKey, Profile.UserLicense.Name, Username, UserRole.Name + FROM User + WHERE Id = :userId + ]; + } + + /** + * @description Creates and inserts a `Group` record for testing queues, using the specified SObject Type + * @param queueDeveloperName The developer name to use for the new queue (stored in `Group.DeveloperName`) + * @param sobjectType The `SObjectType` that the queue should be able to own (stored in `QueueSObject.SObjectType`) + * @return The inserted `Group` record - it is automatically inserted into the database, as well as 1 child `QueueSObject` record. + */ + public static Group insertQueue(String queueDeveloperName, Schema.SObjectType sobjectType) { + Group queue = new Group(DeveloperName = queueDeveloperName, Name = queueDeveloperName, Type = 'Queue'); + insert queue; + + // To avoid a MIXED_DML_OPERATION exception, use System.runs() for inserting the QueueSObject record + System.runAs(new User(Id = UserInfo.getUserId())) { + QueueSObject queueSObject = new QueueSObject(QueueId = queue.Id, SObjectType = sobjectType.getDescribe().getName()); + insert queueSObject; + } + + return queue; + } + + /** + * @description Sets a value for read-only fields that typically cannot be directly set on some SObjects + * @param record The `SObject` record to update + * @param field The `Schema.SObjectField` for the field to update + * @param value The field value to populate on the provied `SObject` record + * @return A new copy of the original `SObject` record that has the specified read-only field populated + */ + public static SObject setReadOnlyField(SObject record, Schema.SObjectField field, Object value) { + return setReadOnlyField(record, new Map{ field => value }); + } + + /** + * @description Sets values for read-only fields that typically cannot be directly set on some SObjects + * @param record record description + * @param changesToFields An instance of `Map containing the read-only fields and corresponding + * field values to populate on the provied `SObject` record + * @return A new copy of the original `SObject` record that has the specified read-only fields populated + */ + public static SObject setReadOnlyField(SObject record, Map changesToFields) { + String serializedRecord = JSON.serialize(record); + Map deserializedRecordMap = (Map) JSON.deserializeUntyped(serializedRecord); + + // Loop through the deserialized record map and put the field & value + // Since it's a map, if the field already exists on the SObject, it's updated (or added if it wasn't there already) + for (Schema.SObjectField sobjectField : changesToFields.keySet()) { + String fieldName = sobjectField.getDescribe().getName(); + deserializedRecordMap.put(fieldName, changesToFields.get(sobjectField)); + } + + serializedRecord = JSON.serialize(deserializedRecordMap); + return (SObject) JSON.deserialize(serializedRecord, SObject.class); + } + + @SuppressWarnings('PMD.ApexDoc') + public class MockBatchableContext implements Database.BatchableContext { + private String childJobId; + private String jobId; + + public MockBatchableContext(String jobId) { + this(jobId, null); + } + + public MockBatchableContext(String jobId, Id childJobId) { + this.jobId = jobId; + this.childJobId = childJobId; + } + + public String getChildJobId() { + return childJobId; + } + + public String getJobId() { + return jobId; + } + } + + @SuppressWarnings('PMD.ApexDoc, PMD.EmptyStatementBlock') + public class MockHttpCallout implements HttpCalloutMock { + public HttpRequest request { get; private set; } + public HttpResponse response { get; private set; } + public String responseBody { get; private set; } + public Integer statusCode { get; private set; } + + public MockHttpCallout() { + } + + public MockHttpCallout setResponseBody(String responseBody) { + this.responseBody = responseBody; + return this; + } + + public MockHttpCallout setStatusCode(Integer statusCode) { + this.statusCode = statusCode; + return this; + } + + public HttpResponse respond(HttpRequest request) { + this.request = request; + + this.response = new HttpResponse(); + if (String.isNotBlank(this.responseBody) == true) { + response.setBody(this.responseBody); + } + response.setStatusCode(this.statusCode); + return response; + } + } + + /** + * @description Class used to create or update an `SObject` record with static fake data. + * This is useful in situations where you need to have fields populated, but the specific + * values used are not relevant to a particular test. + * This class can be used when Apex writing tests for plugins. + */ + public without sharing class SObjectTestDataBuilder { + private final Schema.SObjectType sobjectType; + private final SObject record; + + /** + * @description Creates a new builder instance for the specified `SObjectType`, including creating a + * new `SObject` record. The new `SObject` record is created with any default field values that + * have been configured on the `SObjectType`. + * @param sobjectType The `SObjectType` to use for generating a new test `SObject` record + */ + private SObjectTestDataBuilder(Schema.SObjectType sobjectType) { + this(sobjectType.newSObject(null, true)); + } + + /** + * @description Creates a new builder instance for the specified `SObject` record + * @param record The existing test `SObject` record to populate with sample data + */ + private SObjectTestDataBuilder(SObject record) { + this.record = record; + this.sobjectType = record.getSObjectType(); + + this.loadFields(); + } + + /** + * @description Generates a mock record ID for the builder's `SObject` record + * @return The same instance of `SObjectTestDataBuilder`, useful for chaining methods + */ + public SObjectTestDataBuilder populateMockId() { + this.record.Id = createId(this.sobjectType); + return this; + } + + /** + * @description Sets a value on all editable fields, unless the `SObject` record already had a value specified for a field (including `null`) + * @return The `SObject` record, with all editable fields populated + */ + public SObjectTestDataBuilder populateAllFields() { + this.setUnpopulatedFieldsOnRecord(SOBJECT_TYPE_TO_ALL_FIELDS.get(this.sobjectType)); + return this; + } + + /** + * @description Sets a value on all editable required fields, unless the `SObject` record already had a value specified for a field (including `null`) + * @return The `SObject` record, with all editable required fields populated + */ + public SObjectTestDataBuilder populateRequiredFields() { + this.setUnpopulatedFieldsOnRecord(SOBJECT_TYPE_TO_REQUIRED_FIELDS.get(this.sobjectType)); + return this; + } + + /** + * @description Returns the builder's `SObject` record with fields populated based on + * which builder methods have been called + * @return The builder's `SObject` record that was either provided by the calling code, or generated + * by the builder (depending on which constructor was used for `SObjectTestDataBuilder`) + */ + public SObject getRecord() { + return this.record; + } + + private void loadFields() { + if (SOBJECT_TYPE_TO_ALL_FIELDS.containsKey(this.sobjectType) == true && SOBJECT_TYPE_TO_REQUIRED_FIELDS.containsKey(this.sobjectType) == true) { + return; + } + + SOBJECT_TYPE_TO_ALL_FIELDS.put(this.sobjectType, new List()); + SOBJECT_TYPE_TO_REQUIRED_FIELDS.put(this.sobjectType, new List()); + for (Schema.SObjectField field : this.sobjectType.getDescribe().fields.getMap().values()) { + if (field.getDescribe().isCreateable() == false) { + continue; + } + + SOBJECT_TYPE_TO_ALL_FIELDS.get(this.sobjectType).add(field); + if (field.getDescribe().isNillable() == false) { + // If a field is not nillable & it is createable, then it's required + SOBJECT_TYPE_TO_REQUIRED_FIELDS.get(this.sobjectType).add(field); + } + } + } + + private void setUnpopulatedFieldsOnRecord(List fields) { + Map populatedFields = this.record.getPopulatedFieldsAsMap(); + for (Schema.SObjectField field : fields) { + Schema.DescribeFieldResult fieldDescribe = field.getDescribe(); + // If a field was already populated by using the constructor 'TestDataFactory(SObject record)', then don't change it + if (populatedFields.containsKey(fieldDescribe.getName())) { + continue; + } + + Object fieldValue; + if (fieldDescribe.getDefaultValue() != null) { + // If there is a default value setup for the field, use it + fieldValue = fieldDescribe.getDefaultValue(); + } else { + // Otherwise, we'll generate our own test value to use, based on the field's metadata + fieldValue = this.getTestValue(fieldDescribe); + } + + // If we now have a value to use, set it on the record + if (fieldValue != null) { + this.record.put(field, fieldValue); + } + } + } + + private Object getTestValue(Schema.DescribeFieldResult fieldDescribe) { + // Since Apex does not support case statements, we use several ugly IF-ELSE statements + // Some more complex data types, like ID & Reference, require other objects to be created + // This implementation delegates that responsibility to the test classes since DML is required to get a valid ID, + // but the logic below could be updated to support creating parent objects if needed + + // Unsupported display types have been commented-out below + /* + Schema.DisplayType.Address, Schema.DisplayType.AnyType, Schema.DisplayType.Base64, + Schema.DisplayType.DataCategoryGroupReference, Schema.DisplayType.Id, Schema.DisplayType.Reference + */ + switch on fieldDescribe.getType() { + when Boolean { + return false; + } + when Combobox { + return this.getStringValue(fieldDescribe); + } + when Currency { + return (19.85).setScale(fieldDescribe.getScale()); + } + when Date { + return System.today(); + } + when Datetime { + return System.now(); + } + when Double { + return (3.14).setScale(fieldDescribe.getScale()); + } + when Email { + return 'test@example.com'; + } + when EncryptedString { + return this.getStringValue(fieldDescribe); + } + when Integer { + return 1; + } + when Long { + return 1234567890L; + } + when MultiPicklist { + return fieldDescribe.getPicklistValues().get(0).getValue(); + } + when Percent { + return (0.42).setScale(fieldDescribe.getScale()); + } + when Phone { + return '+34 999 11 22 33'; + } + when Picklist { + return fieldDescribe.getPicklistValues().get(0).getValue(); + } + when String { + return this.getStringValue(fieldDescribe); + } + when TextArea { + return this.getStringValue(fieldDescribe); + } + when Time { + return Time.newInstance(13, 30, 6, 20); + } + when Url { + return 'https://salesforce.com'; + } + when else { + // Any non-supported display types will return null - test classes will need to handle setting the values + return null; + } + } + } + + private String getStringValue(Schema.DescribeFieldResult fieldDescribe) { + String stringValue = 'Test string for ' + fieldDescribe.getType(); + Integer maxLength = fieldDescribe.getLength(); + + return stringValue.left(maxLength).trim(); + } + } +} diff --git a/nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin.cls-meta.xml b/nebula-logger/core/tests/configuration/utilities/LoggerMockDataCreator.cls-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin.cls-meta.xml rename to nebula-logger/core/tests/configuration/utilities/LoggerMockDataCreator.cls-meta.xml diff --git a/nebula-logger/core/tests/log-management/classes/LogBatchPurgeScheduler_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogBatchPurgeScheduler_Tests.cls index 392d9585e..c63b111ac 100644 --- a/nebula-logger/core/tests/log-management/classes/LogBatchPurgeScheduler_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogBatchPurgeScheduler_Tests.cls @@ -4,19 +4,16 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=false) private class LogBatchPurgeScheduler_Tests { private static final String TEST_CRON_EXPRESSION = '0 0 23 * * ?'; @IsTest static void it_should_schedule_job() { - Test.startTest(); + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = 'true')); - LoggerParameter.setMockParameter(new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = 'true')); Id jobId = System.schedule('Test schedule of LogBatchPurgeScheduler', TEST_CRON_EXPRESSION, new LogBatchPurgeScheduler()); - Test.stopTest(); - CronTrigger cronTrigger = [SELECT Id, CronExpression, TimesTriggered, NextFireTime FROM CronTrigger WHERE Id = :jobId]; System.assertEquals(TEST_CRON_EXPRESSION, cronTrigger.CronExpression); System.assertEquals(0, cronTrigger.TimesTriggered); diff --git a/nebula-logger/core/tests/log-management/classes/LogBatchPurger_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogBatchPurger_Tests.cls index d707c4dae..de294434a 100644 --- a/nebula-logger/core/tests/log-management/classes/LogBatchPurger_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogBatchPurger_Tests.cls @@ -4,68 +4,110 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LogBatchPurger_Tests { - private static final Integer NUMBER_OF_LOG_ENTRIES = 10; + private static final Integer NUMBER_OF_LOG_ENTRIES_TO_CREATE = 3; + + private static Boolean ranPluginStart = false; + private static Boolean ranPluginExecute = false; + private static Boolean ranPluginFinish = false; + private static LoggerPlugin__mdt pluginConfiguration; + private static LoggerBatchableContext batchInput; + private static List pluginRecords; @TestSetup static void setupData() { + LoggerSObjectHandler.shouldExecute(false); LoggerSettings__c settings = Logger.getUserSettings(); settings.IsEnabled__c = false; settings.LoggingLevel__c = LoggingLevel.FINEST.name(); insert settings; Date scheduledDeletionDate = System.today().addDays(-7); - Log__c log = new Log__c(LogRetentionDate__c = scheduledDeletionDate, TransactionId__c = '1234'); + Log__c log = new Log__c(LogRetentionDate__c = scheduledDeletionDate); + LoggerMockDataCreator.createDataBuilder(log).populateRequiredFields().getRecord(); insert log; List logEntries = new List(); - for (Integer i = 0; i < NUMBER_OF_LOG_ENTRIES; i++) { + for (Integer i = 0; i < NUMBER_OF_LOG_ENTRIES_TO_CREATE; i++) { LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.INFO.name()); - + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); logEntries.add(logEntry); } insert logEntries; - LoggerTag__c tag = new LoggerTag__c(Name = 'My important tag'); + LoggerTag__c tag = (LoggerTag__c) LoggerMockDataCreator.createDataBuilder(Schema.LoggerTag__c.SObjectType).populateRequiredFields().getRecord(); insert tag; List logEntryTags = new List(); for (LogEntry__c logEntry : logEntries) { LogEntryTag__c logEntryTag = new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id); + LoggerMockDataCreator.createDataBuilder(logEntryTag).populateRequiredFields().getRecord(); logEntryTags.add(logEntryTag); } insert logEntryTags; } + @IsTest + static void it_should_throw_an_exception_when_data_cannot_be_purged() { + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = 'true')); + LogBatchPurger batchJobInstance = new LogBatchPurger(); + batchJobInstance.currentSObjectType = Schema.Log__c.SObjectType; + List nullLogsList = null; + Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + Exception thrownNullPointerException; + + try { + batchJobInstance.execute(mockBatchableContext, nullLogsList); + } catch (NullPointerException ex) { + thrownNullPointerException = ex; + } + + System.assertNotEquals(null, thrownNullPointerException); + } + + @IsTest + static void it_should_default_chained_job_batch_size_to_200() { + LogBatchPurger batchJobInstance = new LogBatchPurger(); + + System.assertEquals(200, batchJobInstance.chainedBatchSize); + } + + @IsTest + static void it_should_set_chained_job_batch_size() { + LogBatchPurger batchJobInstance = new LogBatchPurger(); + Integer specifiedBatchSize = 123; + + batchJobInstance.setChainedBatchSize(specifiedBatchSize); + + System.assertEquals(specifiedBatchSize, batchJobInstance.chainedBatchSize); + } + @IsTest static void it_should_delete_a_log_after_scheduled_deletion_date_when_system_messages_enabled() { LoggerSettings__c settings = Logger.getUserSettings(); settings.IsEnabled__c = true; settings.LoggingLevel__c = LoggingLevel.FINEST.name(); upsert settings; - - LoggerParameter.setMockParameter(new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = 'true')); - + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = 'true')); System.assertEquals(true, Logger.getUserSettings().IsEnabled__c); System.assertEquals(true, LoggerParameter.getBoolean('EnableLoggerSystemMessages', null)); System.assertEquals(LoggingLevel.FINEST.name(), Logger.getUserSettings().LoggingLevel__c); - List logs = [SELECT Id, LogRetentionDate__c FROM Log__c]; List logEntries = [SELECT Id FROM LogEntry__c]; - System.assertEquals(1, logs.size()); - System.assertEquals(NUMBER_OF_LOG_ENTRIES, logEntries.size()); - + System.assertEquals(NUMBER_OF_LOG_ENTRIES_TO_CREATE, logEntries.size()); // Verify assumption that the log in the database has a deletion date in the past Log__c log = logs.get(0); System.assertNotEquals(null, log.LogRetentionDate__c); System.assert(log.LogRetentionDate__c < System.today()); - Test.startTest(); + LoggerSObjectHandler.shouldExecute(true); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + Database.executeBatch(new LogBatchPurger()); - Test.stopTest(); + Test.stopTest(); logs = [SELECT Id FROM Log__c WHERE Id IN :logs]; logEntries = [SELECT Id FROM LogEntry__c WHERE Id IN :logEntries]; System.assertEquals(0, logs.size(), logs); @@ -77,27 +119,25 @@ private class LogBatchPurger_Tests { LoggerSettings__c settings = Logger.getUserSettings(); settings.IsEnabled__c = true; upsert settings; - LoggerParameter.setMockParameter(new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = String.valueOf(false))); - + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'EnableLoggerSystemMessages', Value__c = String.valueOf(false))); System.assertEquals(true, Logger.getUserSettings().IsEnabled__c); System.assertEquals(false, LoggerParameter.getBoolean('EnableLoggerSystemMessages', null)); System.assertEquals(LoggingLevel.FINEST.name(), Logger.getUserSettings().LoggingLevel__c); - List logs = [SELECT Id, LogRetentionDate__c FROM Log__c]; List logEntries = [SELECT Id FROM LogEntry__c]; - System.assertEquals(1, logs.size()); - System.assertEquals(NUMBER_OF_LOG_ENTRIES, logEntries.size()); - + System.assertEquals(NUMBER_OF_LOG_ENTRIES_TO_CREATE, logEntries.size()); // Verify assumption that the log in the database has a deletion date in the past Log__c log = logs.get(0); System.assertNotEquals(null, log.LogRetentionDate__c); System.assert(log.LogRetentionDate__c < System.today()); - Test.startTest(); + LoggerSObjectHandler.shouldExecute(false); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + Database.executeBatch(new LogBatchPurger()); - Test.stopTest(); + Test.stopTest(); logs = [SELECT Id FROM Log__c WHERE Id IN :logs]; logEntries = [SELECT Id FROM LogEntry__c WHERE Id IN :logEntries]; System.assertEquals(0, logs.size(), logs); @@ -106,57 +146,60 @@ private class LogBatchPurger_Tests { @IsTest static void it_should_not_delete_a_log_before_scheduled_deletion_date() { + LoggerSObjectHandler.shouldExecute(false); List logs = [SELECT Id, LogRetentionDate__c FROM Log__c]; List logEntries = [SELECT Id FROM LogEntry__c]; - System.assertEquals(1, logs.size()); - System.assertEquals(NUMBER_OF_LOG_ENTRIES, logEntries.size()); - + System.assertEquals(NUMBER_OF_LOG_ENTRIES_TO_CREATE, logEntries.size()); // Set the log's deletion date to be in the future Log__c log = logs.get(0); log.LogRetentionDate__c = System.today().addDays(7); update log; log = [SELECT Id, LogRetentionDate__c FROM Log__c]; System.assert(log.LogRetentionDate__c > System.today()); - Test.startTest(); + LoggerSObjectHandler.shouldExecute(false); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + Database.executeBatch(new LogBatchPurger()); - Test.stopTest(); + Test.stopTest(); logs = [SELECT Id FROM Log__c WHERE Id IN :logs]; logEntries = [SELECT Id FROM LogEntry__c WHERE Id IN :logEntries]; System.assertEquals(1, logs.size()); - System.assertEquals(NUMBER_OF_LOG_ENTRIES, logEntries.size()); + System.assertEquals(NUMBER_OF_LOG_ENTRIES_TO_CREATE, logEntries.size()); } @IsTest static void it_should_not_delete_a_log_without_a_scheduled_deletion_date_and_with_log_entries() { + LoggerSObjectHandler.shouldExecute(false); List logs = [SELECT Id, TotalLogEntries__c, LogRetentionDate__c FROM Log__c]; List logEntries = [SELECT Id FROM LogEntry__c]; - System.assertEquals(1, logs.size()); - System.assertEquals(NUMBER_OF_LOG_ENTRIES, logs.get(0).TotalLogEntries__c); - System.assertEquals(NUMBER_OF_LOG_ENTRIES, logEntries.size()); - + System.assertEquals(NUMBER_OF_LOG_ENTRIES_TO_CREATE, logs.get(0).TotalLogEntries__c); + System.assertEquals(NUMBER_OF_LOG_ENTRIES_TO_CREATE, logEntries.size()); // Set the log's deletion date to be null Log__c log = logs.get(0); log.LogRetentionDate__c = null; update log; log = [SELECT Id, LogRetentionDate__c FROM Log__c]; System.assertEquals(null, log.LogRetentionDate__c); - Test.startTest(); + LoggerSObjectHandler.shouldExecute(false); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + Database.executeBatch(new LogBatchPurger()); - Test.stopTest(); + Test.stopTest(); logs = [SELECT Id FROM Log__c WHERE Id IN :logs]; logEntries = [SELECT Id FROM LogEntry__c WHERE Id IN :logEntries]; System.assertEquals(1, logs.size(), logs); - System.assertEquals(NUMBER_OF_LOG_ENTRIES, logEntries.size(), logEntries); + System.assertEquals(NUMBER_OF_LOG_ENTRIES_TO_CREATE, logEntries.size(), logEntries); } @IsTest static void it_should_delete_a_log_with_a_scheduled_deletion_date_and_without_log_entries() { + LoggerSObjectHandler.shouldExecute(false); Date retentionDate = System.today().addDays(-1); Log__c log = new Log__c(TransactionId__c = '5678'); insert log; @@ -165,17 +208,20 @@ private class LogBatchPurger_Tests { log = [SELECT Id, LogRetentionDate__c, TotalLogEntries__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(retentionDate, log.LogRetentionDate__c, 'Log should not have a retention date'); System.assertEquals(0, log.TotalLogEntries__c, 'Log should not have any related log entries'); - Test.startTest(); + LoggerSObjectHandler.shouldExecute(false); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + Database.executeBatch(new LogBatchPurger()); - Test.stopTest(); + Test.stopTest(); List matchingLogs = [SELECT Id FROM Log__c WHERE Id = :log.Id]; System.assertEquals(0, matchingLogs.size(), 'Test log should have been deleted'); } @IsTest static void it_should_delete_a_log_without_a_scheduled_deletion_date_and_without_log_entries() { + LoggerSObjectHandler.shouldExecute(false); Log__c log = new Log__c(TransactionId__c = '5678'); insert log; log.LogRetentionDate__c = null; @@ -183,12 +229,191 @@ private class LogBatchPurger_Tests { log = [SELECT Id, LogRetentionDate__c, TotalLogEntries__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(null, log.LogRetentionDate__c, 'Log should not have a retention date'); System.assertEquals(0, log.TotalLogEntries__c, 'Log should not have any related log entries'); - Test.startTest(); + LoggerSObjectHandler.shouldExecute(false); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + Database.executeBatch(new LogBatchPurger()); - Test.stopTest(); + Test.stopTest(); List matchingLogs = [SELECT Id FROM Log__c WHERE Id = :log.Id]; System.assertEquals(0, matchingLogs.size(), 'Test log should have been deleted'); } + + @IsTest + static void it_should_run_apex_plugin_in_start_method() { + LoggerSObjectHandler.shouldExecute(false); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'Mock_Plugin', + IsEnabled__c = true, + BatchPurgerApexClass__c = MockBatchPurgerPlugin.class.getName() + ); + LoggerTestConfigurator.setMock(mockPluginConfiguration); + LogBatchPurger batchJobInstance = new LogBatchPurger(); + batchJobInstance.currentSObjectType = Schema.LogEntry__c.SObjectType; + System.assertEquals(false, ranPluginStart); + System.assertEquals(false, ranPluginExecute); + System.assertEquals(false, ranPluginFinish); + System.assertEquals(null, pluginConfiguration); + System.assertEquals(null, batchInput); + Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + + batchJobInstance.start(mockBatchableContext); + + LoggerBatchableContext expectedInput = new LoggerBatchableContext(mockBatchableContext, batchJobInstance.currentSObjectType); + System.assertEquals( + 3, + batchJobInstance.getExecutedApexPlugins().size(), + 'The map of executed Apex plugins should have 3 keys - one for each enum value in LogBatchPurger.BatchableMethod (START, EXECUTE, and FINISH)' + ); + System.assertEquals( + 1, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.START).size(), + 'One Apex plugin should have run in the batch job\'s start method' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).size(), + 'No Apex plugins should have run in the batch job\'s execute method' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.FINISH).size(), + 'No Apex plugins should have run in the batch job\'s finish method' + ); + LoggerPlugin.Batchable apexStartPlugin = batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.START).get(0); + System.assertEquals(true, apexStartPlugin instanceof MockBatchPurgerPlugin, apexStartPlugin); + System.assertEquals(true, ranPluginStart); + System.assertEquals(false, ranPluginExecute); + System.assertEquals(false, ranPluginFinish); + System.assertEquals(mockPluginConfiguration, pluginConfiguration); + System.assertEquals(expectedInput.batchableContext, batchInput.batchableContext); + System.assertEquals(expectedInput.sobjectType, batchInput.sobjectType); + } + + @IsTest + static void it_should_run_apex_plugin_in_execute_method() { + LoggerSObjectHandler.shouldExecute(false); + LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'Mock_Plugin', + IsEnabled__c = true, + BatchPurgerApexClass__c = MockBatchPurgerPlugin.class.getName() + ); + LoggerTestConfigurator.setMock(mockPluginConfiguration); + LogBatchPurger batchJobInstance = new LogBatchPurger(); + batchJobInstance.currentSObjectType = Schema.LogEntry__c.SObjectType; + System.assertEquals(false, ranPluginStart); + System.assertEquals(false, ranPluginExecute); + System.assertEquals(false, ranPluginFinish); + System.assertEquals(null, pluginConfiguration); + System.assertEquals(null, batchInput); + List logsToDelete = [SELECT Id FROM Log__c]; + System.assertNotEquals(0, logsToDelete.size()); + Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + + batchJobInstance.execute(mockBatchableContext, logsToDelete); + + LoggerBatchableContext expectedInput = new LoggerBatchableContext(mockBatchableContext, batchJobInstance.currentSObjectType); + System.assertEquals( + 3, + batchJobInstance.getExecutedApexPlugins().size(), + 'The map of executed Apex plugins should have 3 keys - one for each enum value in LogBatchPurger.BatchableMethod (START, EXECUTE, and FINISH)' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.START).size(), + 'No Apex plugins should have run in the batch job\'s start method' + ); + System.assertEquals( + 1, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).size(), + 'One Apex plugin should have run in the batch job\'s execute method' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.FINISH).size(), + 'No Apex plugins should have run in the batch job\'s finish method' + ); + LoggerPlugin.Batchable apexExecutePlugin = batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).get(0); + System.assertEquals(true, apexExecutePlugin instanceof MockBatchPurgerPlugin, apexExecutePlugin); + System.assertEquals(false, ranPluginStart); + System.assertEquals(true, ranPluginExecute); + System.assertEquals(false, ranPluginFinish); + System.assertEquals(mockPluginConfiguration, pluginConfiguration); + System.assertEquals(expectedInput.batchableContext, batchInput.batchableContext); + logsToDelete = [SELECT Id FROM Log__c WHERE Id IN :logsToDelete]; + System.assertEquals(0, logsToDelete.size(), 'All logs should have still been deleted by LogBatchPurger after running plugins: ' + logsToDelete); + } + + @IsTest + static void it_should_run_apex_plugin_in_finish_method() { + LoggerSObjectHandler.shouldExecute(false); + LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'Mock_Plugin', + IsEnabled__c = true, + BatchPurgerApexClass__c = MockBatchPurgerPlugin.class.getName() + ); + LoggerTestConfigurator.setMock(mockPluginConfiguration); + LogBatchPurger batchJobInstance = new LogBatchPurger(); + batchJobInstance.currentSObjectType = Schema.LogEntry__c.SObjectType; + System.assertEquals(false, ranPluginStart); + System.assertEquals(false, ranPluginExecute); + System.assertEquals(false, ranPluginFinish); + System.assertEquals(null, pluginConfiguration); + System.assertEquals(null, batchInput); + Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + + batchJobInstance.finish(mockBatchableContext); + + LoggerBatchableContext expectedInput = new LoggerBatchableContext(mockBatchableContext, batchJobInstance.currentSObjectType); + System.assertEquals( + 3, + batchJobInstance.getExecutedApexPlugins().size(), + 'The map of executed Apex plugins should have 3 keys - one for each enum value in LogBatchPurger.BatchableMethod (START, EXECUTE, and FINISH)' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.START).size(), + 'No Apex plugins should have run in the batch job\'s start method' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).size(), + 'No Apex plugins should have run in the batch job\'s execute method' + ); + System.assertEquals( + 1, + batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.FINISH).size(), + 'One Apex plugin should have run in the batch job\'s finish method' + ); + LoggerPlugin.Batchable apexFinishPlugin = batchJobInstance.getExecutedApexPlugins().get(LogBatchPurger.BatchableMethod.FINISH).get(0); + System.assertEquals(true, apexFinishPlugin instanceof MockBatchPurgerPlugin, apexFinishPlugin); + System.assertEquals(false, ranPluginStart); + System.assertEquals(false, ranPluginExecute); + System.assertEquals(true, ranPluginFinish); + System.assertEquals(mockPluginConfiguration, pluginConfiguration); + System.assertEquals(expectedInput.sobjectType, batchInput.sobjectType); + } + + public class MockBatchPurgerPlugin implements LoggerPlugin.Batchable { + public void start(LoggerPlugin__mdt configuration, LoggerBatchableContext input) { + ranPluginStart = true; + pluginConfiguration = configuration; + batchInput = input; + } + + public void execute(LoggerPlugin__mdt configuration, LoggerBatchableContext input, List scope) { + ranPluginExecute = true; + pluginConfiguration = configuration; + batchInput = input; + pluginRecords = scope; + } + + public void finish(LoggerPlugin__mdt configuration, LoggerBatchableContext input) { + ranPluginFinish = true; + pluginConfiguration = configuration; + batchInput = input; + } + } } diff --git a/nebula-logger/core/tests/log-management/classes/LogEntryEventHandler_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogEntryEventHandler_Tests.cls index f1240d83e..154a3dad5 100644 --- a/nebula-logger/core/tests/log-management/classes/LogEntryEventHandler_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogEntryEventHandler_Tests.cls @@ -4,55 +4,85 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LogEntryEventHandler_Tests { private static final String MOCK_RELEASE_NUMBER = '230.12.2'; private static final String MOCK_RELEASE_VERSION = 'Spring \'21 Patch 12.2'; @IsTest static void it_should_return_the_logEntryEvent_sobjectType() { - Test.startTest(); System.assertEquals(Schema.LogEntryEvent__e.SObjectType, new LogEntryEventHandler().getSObjectType()); - Test.stopTest(); + } + + @IsTest + static void it_should_not_run_when_disabled_via_configuration() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntryEvent__e.SObjectType).IsEnabled__c = false; + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + + LoggerDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + + System.assertEquals( + 0, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should not have executed' + ); } @IsTest static void it_should_gracefully_skip_execution_when_logEntryEvents_list_is_empty() { - List logEntryEvents = new List(); - System.assert(logEntryEvents.isEmpty()); + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; + Integer countOfLogs = [SELECT COUNT() FROM Log__c]; + System.assertEquals(0, countOfLogs); + Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; + System.assertEquals(0, countOfLogEntries); - Test.startTest(); - new LogEntryEventHandler().executeBeforeInsert(logEntryEvents); - new LogEntryEventHandler().executeAfterInsert(logEntryEvents); - Test.stopTest(); + List saveResults = LoggerMockDataStore.getEventBus().publishRecords(new List()); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + + System.assertEquals(true, saveResults.isEmpty()); + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); + countOfLogs = [SELECT COUNT() FROM Log__c]; + System.assertEquals(0, countOfLogs); + countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; + System.assertEquals(0, countOfLogEntries); } @IsTest static void it_should_not_create_log_or_log_entry_data_when_platform_event_storage_is_disabled() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; Integer countOfLogs = [SELECT COUNT() FROM Log__c]; System.assertEquals(0, countOfLogs); Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; System.assertEquals(0, countOfLogEntries); - - Logger.getUserSettings().IsSavingEnabled__c = true; LoggerSettings__c settings = Logger.getUserSettings(); - settings.IsPlatformEventStorageEnabled__c = false; + settings.IsEnabled__c = true; + settings.IsSavingEnabled__c = true; + settings.DefaultPlatformEventStorageLocation__c = null; upsert settings; - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - LoggedById__c = UserInfo.getUserId(), - Message__c = 'my message', - Timestamp__c = System.now(), - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); - Test.startTest(); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); countOfLogs = [SELECT COUNT() FROM Log__c]; System.assertEquals(0, countOfLogs); countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; @@ -60,165 +90,61 @@ private class LogEntryEventHandler_Tests { } @IsTest - static void it_should_normalize_simple_event_data_into_log_and_log_entry() { - String transactionId = '123-456-789-0'; + static void it_should_normalize_event_data_into_log_and_log_entry() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - LoggedById__c = UserInfo.getUserId(), - Message__c = 'my message', - Timestamp__c = System.now(), - TransactionEntryNumber__c = 1, - TransactionId__c = transactionId - ); - - Test.startTest(); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); validateLogFields(logEntryEvent, log); validateLogEntryFields(logEntryEvent, logEntry); } - @IsTest - static void it_should_normalize_complex_event_data_into_log_and_log_entry() { - String transactionId = '123-456-789-0'; - User currentUser = getCurrentUser(); - DmlException dmlException = new DmlException(); - - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - ApiVersion__c = '99.0', - EpochTimestamp__c = System.now().getTime(), - ExceptionStackTrace__c = dmlException.getStackTraceString(), - ExceptionType__c = dmlException.getTypeName(), - LimitsAggregateQueriesMax__c = 987, - LimitsAggregateQueriesUsed__c = 123, - LimitsAsyncCallsMax__c = 987, - LimitsAsyncCallsUsed__c = 123, - LimitsCalloutsMax__c = 987, - LimitsCalloutsUsed__c = 123, - LimitsCpuTimeMax__c = 987, - LimitsCpuTimeUsed__c = 123, - LimitsDmlRowsMax__c = 987, - LimitsDmlRowsUsed__c = 123, - LimitsDmlStatementsMax__c = 987, - LimitsDmlStatementsUsed__c = 123, - LimitsEmailInvocationsMax__c = 987, - LimitsEmailInvocationsUsed__c = 123, - LimitsFutureCallsMax__c = 987, - LimitsFutureCallsUsed__c = 123, - LimitsHeapSizeMax__c = 987, - LimitsHeapSizeUsed__c = 123, - LimitsMobilePushApexCallsMax__c = 987, - LimitsMobilePushApexCallsUsed__c = 123, - LimitsPublishImmediateDmlStatementsMax__c = 987, - LimitsPublishImmediateDmlStatementsUsed__c = 123, - LimitsQueueableJobsMax__c = 987, - LimitsQueueableJobsUsed__c = 123, - LimitsSoqlQueriesMax__c = 987, - LimitsSoqlQueriesUsed__c = 123, - LimitsSoqlQueryLocatorRowsMax__c = 987, - LimitsSoqlQueryLocatorRowsUsed__c = 123, - LimitsSoqlQueryRowsMax__c = 987, - LimitsSoqlQueryRowsUsed__c = 123, - LimitsSoslSearchesMax__c = 987, - LimitsSoslSearchesUsed__c = 123, - Locale__c = 'es-ES', - LoggerVersionNumber__c = 'v4.6.10', - LoggedById__c = UserInfo.getUserId(), - LoggingLevel__c = 'DEBUG', - LoggingLevelOrdinal__c = 9, - LoginDomain__c = 'https://fake.my.salesforce.com', - LoginHistoryId__c = '255.255.255.255', - LoginType__c = null, - LogoutUrl__c = null, - Message__c = 'My message', - MessageTruncated__c = false, - NetworkId__c = 'Fake-ID', - OriginLocation__c = 'right here', - OriginType__c = 'Apex', - ProfileId__c = currentUser.ProfileId, - RecordId__c = currentUser.Id, - RecordJson__c = JSON.serializePretty(currentUser), - RecordSObjectClassification__c = 'Standard Object', - RecordSObjectType__c = 'User', - Scenario__c = 'Apex test execution', - SessionId__c = null, - SessionSecurityLevel__c = null, - SessionType__c = null, - SourceIp__c = null, - StackTrace__c = 'I started here \nand then went there', - SystemMode__c = System.Quiddity.SYNCHRONOUS.name(), - ThemeDisplayed__c = 'SomeTheme', - Timestamp__c = System.now(), - TimeZoneId__c = 'fake-id', - TimeZoneName__c = 'My time zone name', - TransactionEntryNumber__c = 1, - TransactionId__c = transactionId, - TriggerIsExecuting__c = false, - TriggerOperationType__c = 'SomeOperation', - TriggerSObjectType__c = 'Account', - UserLoggingLevel__c = 'DEBUG', - UserLoggingLevelOrdinal__c = 9, - UserRoleId__c = currentUser.UserRoleId, - UserType__c = null - ); - - Test.startTest(); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); - - System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - - Log__c log = getLog(); - System.assertEquals(1, log.LogEntries__r.size()); - LogEntry__c logEntry = log.LogEntries__r.get(0); - - validateLogFields(logEntryEvent, log); - validateLogEntryFields(logEntryEvent, logEntry); - } - @IsTest static void it_should_normalize_event_data_for_multiple_transactions_into_multiple_logs() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; List transactionIds = new List{ '123-456', '789-0' }; System.assertEquals(2, transactionIds.size()); - List logEntryEvents = new List(); for (Integer i = 0; i < transactionIds.size(); i++) { - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - TransactionEntryNumber__c = i, - TransactionId__c = transactionIds.get(i) - ); - + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + logEntryEvent.TransactionEntryNumber__c = i; + logEntryEvent.TransactionId__c = transactionIds.get(i); logEntryEvents.add(logEntryEvent); } System.assertEquals(transactionIds.size(), logEntryEvents.size()); - Test.startTest(); - List saveResults = EventBus.publish(logEntryEvents); - Test.getEventBus().deliver(); - System.assertEquals(transactionIds.size(), saveResults.size()); - Test.stopTest(); + List saveResults = LoggerMockDataStore.getEventBus().publishRecords(logEntryEvents); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.assertEquals(transactionIds.size(), saveResults.size()); for (Database.SaveResult saveResult : saveResults) { System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); } - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); List logs = [SELECT Id, TransactionId__c, (SELECT Id FROM LogEntries__r) FROM Log__c]; - System.assertEquals(2, logs.size(), logs); - - // Convert the list of txn IDs to a set so we can use `contains()` Set uniqueTransactionIds = new Set(transactionIds); for (Log__c log : logs) { System.assert(uniqueTransactionIds.contains(log.TransactionId__c), log.TransactionId__c); @@ -268,32 +194,30 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_create_tag_records_when_enabled() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; System.assertEquals(true, LoggerParameter.TAGGING_IS_ENABLED, 'Tagging is not enabled within test context, cannot execute tagging test'); - List tags = new List{ 'test-tag-1', 'test-tag-2' }; - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - Tags__c = String.join(tags, '\n'), - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + logEntryEvent.Tags__c = String.join(tags, '\n'); - Test.startTest(); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); validateLogFields(logEntryEvent, log); validateLogEntryFields(logEntryEvent, logEntry); - List logEntryTags = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; System.assertEquals(tags.size(), logEntryTags.size()); Set tagsSet = new Set(tags); @@ -304,35 +228,32 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_reuse_existing_tag_records() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; System.assertEquals(true, LoggerParameter.TAGGING_IS_ENABLED, 'Tagging is not enabled within test context, cannot execute tagging test'); - String testTagName = 'Some tag!'; LoggerTag__c tag = new LoggerTag__c(Name = testTagName); insert tag; + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + logEntryEvent.Tags__c = testTagName; - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - Tags__c = testTagName, - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); - - Test.startTest(); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); validateLogFields(logEntryEvent, log); validateLogEntryFields(logEntryEvent, logEntry); - Integer countOfTagsWithTagName = [SELECT COUNT() FROM LoggerTag__c WHERE Name = :testTagName]; System.assertEquals(1, countOfTagsWithTagName); LogEntryTag__c logEntryTag = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; @@ -342,20 +263,16 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_append_tags_for_contains_rule() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; System.assertEquals(true, LoggerParameter.TAGGING_IS_ENABLED, 'Tagging is not enabled within test context, cannot execute tagging test'); - - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - Tags__c = null, - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); - - Test.startTest(); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + logEntryEvent.Message__c = 'my message'; + logEntryEvent.Tags__c = null; String configuredTagName = 'CMDT Tag'; - LogEntryEventHandler.TAG_ASSIGNMENT_RULES.add( + LoggerTestConfigurator.setMock( new LogEntryTagRule__mdt( SObjectField__c = Schema.LogEntry__c.Message__c.getDescribe().getName(), ComparisonType__c = 'CONTAINS', @@ -363,19 +280,22 @@ private class LogEntryEventHandler_Tests { Tags__c = configuredTagName ) ); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); - System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); validateLogFields(logEntryEvent, log); validateLogEntryFields(logEntryEvent, logEntry); - Integer countOfTagsWithTagName = [SELECT COUNT() FROM LoggerTag__c WHERE Name = :configuredTagName]; System.assertEquals(1, countOfTagsWithTagName); LogEntryTag__c logEntryTag = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; @@ -384,20 +304,15 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_append_tags_for_equals_rule() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; System.assertEquals(true, LoggerParameter.TAGGING_IS_ENABLED, 'Tagging is not enabled within test context, cannot execute tagging test'); - - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - Tags__c = null, - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); - - Test.startTest(); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + logEntryEvent.Tags__c = null; String configuredTagName = 'CMDT Tag'; - LogEntryEventHandler.TAG_ASSIGNMENT_RULES.add( + LoggerTestConfigurator.setMock( new LogEntryTagRule__mdt( SObjectField__c = Schema.LogEntry__c.Message__c.getDescribe().getName(), ComparisonType__c = 'EQUALS', @@ -405,19 +320,22 @@ private class LogEntryEventHandler_Tests { Tags__c = configuredTagName ) ); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); - System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); validateLogFields(logEntryEvent, log); validateLogEntryFields(logEntryEvent, logEntry); - Integer countOfTagsWithTagName = [SELECT COUNT() FROM LoggerTag__c WHERE Name = :configuredTagName]; System.assertEquals(1, countOfTagsWithTagName); LogEntryTag__c logEntryTag = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; @@ -426,22 +344,17 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_append_tags_for_regex_rule() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; System.assertEquals(true, LoggerParameter.TAGGING_IS_ENABLED, 'Tagging is not enabled within test context, cannot execute tagging test'); - String zipCodeRegEx = '(^[0-9]{4}?[0-9]$|^[0-9]{4}?[0-9]-[0-9]{4}$)'; - - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = '94541', - Timestamp__c = System.now(), - Tags__c = null, - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); - - Test.startTest(); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + logEntryEvent.Message__c = '94541'; + logEntryEvent.Tags__c = null; String configuredTagName = 'CMDT Tag'; - LogEntryEventHandler.TAG_ASSIGNMENT_RULES.add( + LoggerTestConfigurator.setMock( new LogEntryTagRule__mdt( SObjectField__c = Schema.LogEntry__c.Message__c.getDescribe().getName(), ComparisonType__c = 'MATCHES_REGEX', @@ -449,19 +362,22 @@ private class LogEntryEventHandler_Tests { Tags__c = configuredTagName ) ); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); - System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); validateLogFields(logEntryEvent, log); validateLogEntryFields(logEntryEvent, logEntry); - Integer countOfTagsWithTagName = [SELECT COUNT() FROM LoggerTag__c WHERE Name = :configuredTagName]; System.assertEquals(1, countOfTagsWithTagName); LogEntryTag__c logEntryTag = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; @@ -470,20 +386,16 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_append_tags_for_startsWith_rule() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; System.assertEquals(true, LoggerParameter.TAGGING_IS_ENABLED, 'Tagging is not enabled within test context, cannot execute tagging test'); - - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - Tags__c = null, - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); - - Test.startTest(); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); + logEntryEvent.Message__c = 'my message'; + logEntryEvent.Tags__c = null; String configuredTagName = 'CMDT Tag'; - LogEntryEventHandler.TAG_ASSIGNMENT_RULES.add( + LoggerTestConfigurator.setMock( new LogEntryTagRule__mdt( SObjectField__c = Schema.LogEntry__c.Message__c.getDescribe().getName(), ComparisonType__c = 'STARTS_WITH', @@ -491,19 +403,22 @@ private class LogEntryEventHandler_Tests { Tags__c = configuredTagName ) ); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); - System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); validateLogFields(logEntryEvent, log); validateLogEntryFields(logEntryEvent, logEntry); - Integer countOfTagsWithTagName = [SELECT COUNT() FROM LoggerTag__c WHERE Name = :configuredTagName]; System.assertEquals(1, countOfTagsWithTagName); LogEntryTag__c logEntryTag = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; @@ -512,37 +427,32 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_set_api_release_number_and_api_release_version_from_callout_when_enabled() { - Test.setMock(HttpCalloutMock.class, new SuccessCalloutMock()); + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; + System.Test.setMock(HttpCalloutMock.class, LoggerMockDataCreator.createHttpCallout().setStatusCode(200).setResponseBody(createStatusApiResponseJson())); LoggerParameter__mdt mockCallStatusApiParameter = new LoggerParameter__mdt(DeveloperName = 'CallStatusApi', Value__c = 'true'); - LoggerParameter.setMockParameter(mockCallStatusApiParameter); + LoggerTestConfigurator.setMock(mockCallStatusApiParameter); System.assertEquals(true, LoggerParameter.CALL_STATUS_API); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - TransactionEntryNumber__c = 1, - TransactionId__c = '123-456-789-0' - ); - - Database.SaveResult saveResult; - - Test.startTest(); - - saveResult = EventBus.publish(logEntryEvent); - // Normally, you don't have to call Test.getEventBus().deliver() if you're also using Test.stopTest() - // But, in this case, there are 3 transactions happening (original test, async platform event, and async future method) - // So, call Test.getEventBus().deliver() and Test.stopTest() to make sure all transactions are completed before running asserts - Test.getEventBus().deliver(); - - Test.stopTest(); + // TODO Test.startTest() & stopTest() are still being used so that the callout in LogEntryEventHandler executes, + // but it be nice if this was instead mocked via LoggerMockDataStore (? or some other similar class?) + System.Test.startTest(); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.Test.stopTest(); System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(MOCK_RELEASE_NUMBER, log.ApiReleaseNumber__c); System.assertEquals(MOCK_RELEASE_VERSION, log.ApiReleaseVersion__c); validateLogFields(logEntryEvent, log); @@ -551,44 +461,35 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_set_api_release_number_and_api_release_version_from_recent_record() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; Log__c recentLog = new Log__c(ApiReleaseNumber__c = 'QWERTY', ApiReleaseVersion__c = 'ASDF', TransactionId__c = 'ABC-XYZ'); insert recentLog; - insert new LogEntry__c(Log__c = recentLog.Id, Timestamp__c = System.now().addHours(-1)); - LoggerParameter__mdt mockCallStatusApiParameter = new LoggerParameter__mdt(DeveloperName = 'CallStatusApi', Value__c = 'true'); - LoggerParameter.setMockParameter(mockCallStatusApiParameter); + LoggerTestConfigurator.setMock(mockCallStatusApiParameter); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); - String transactionId = '123-456-789-0'; - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - TransactionEntryNumber__c = 1, - TransactionId__c = transactionId - ); - - Database.SaveResult saveResult; - - Test.startTest(); - - saveResult = EventBus.publish(logEntryEvent); - // Normally, you don't have to call Test.getEventBus().deliver() if you're also using Test.stopTest() - // But, in this case, there are 3 transactions happening (original test, async platform event, and async future method) - // So, call Test.getEventBus().deliver() and Test.stopTest() to make sure all transactions are completed before running asserts - Test.getEventBus().deliver(); - - Test.stopTest(); + // TODO Test.startTest() & stopTest() are still being used so that the callout in LogEntryEventHandler executes, + // but it be nice if this was instead mocked via LoggerMockDataStore (? or some other similar class?) + System.Test.startTest(); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.Test.stopTest(); System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); System.assertEquals(0, Limits.getCallouts()); - Log__c log = getLog(); System.assertNotEquals(recentLog.Id, log.Id, log.StartTime__c); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(recentLog.ApiReleaseNumber__c, log.ApiReleaseNumber__c); System.assertEquals(recentLog.ApiReleaseVersion__c, log.ApiReleaseVersion__c); validateLogFields(logEntryEvent, log); @@ -597,29 +498,32 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_set_not_api_release_number_or_api_release_version_from_callout_when_disabled() { - Test.setMock(HttpCalloutMock.class, new SuccessCalloutMock()); + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; + Test.setMock(HttpCalloutMock.class, LoggerMockDataCreator.createHttpCallout().setStatusCode(400).setResponseBody(createStatusApiResponseJson())); LoggerParameter__mdt mockCallStatusApiParameter = new LoggerParameter__mdt(DeveloperName = 'CallStatusApi', Value__c = 'false'); - LoggerParameter.setMockParameter(mockCallStatusApiParameter); + LoggerTestConfigurator.setMock(mockCallStatusApiParameter); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); - String transactionId = '123-456-789-0'; - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - EpochTimestamp__c = System.now().getTime(), - Message__c = 'my message', - Timestamp__c = System.now(), - TransactionEntryNumber__c = 1, - TransactionId__c = transactionId - ); - - Test.startTest(); - Database.SaveResult saveResult = EventBus.publish(logEntryEvent); - Test.stopTest(); + // TODO Test.startTest() & stopTest() are still being used so that the callout in LogEntryEventHandler executes, + // but it be nice if this was instead mocked via LoggerMockDataStore (? or some other similar class?) + System.Test.startTest(); + Database.SaveResult saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.Test.stopTest(); System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); + System.assertEquals(0, Limits.getCallouts()); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); System.assertEquals(null, log.ApiReleaseNumber__c); System.assertEquals(null, log.ApiReleaseVersion__c); @@ -629,43 +533,39 @@ private class LogEntryEventHandler_Tests { @IsTest static void it_should_skip_setting_api_release_number_and_api_release_version_when_callout_fails() { - Test.setMock(HttpCalloutMock.class, new FailureCalloutMock()); + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.Log__c.SObjectType).IsEnabled__c = false; + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; + Test.setMock(HttpCalloutMock.class, LoggerMockDataCreator.createHttpCallout().setStatusCode(400).setResponseBody(createStatusApiResponseJson())); LoggerParameter__mdt mockCallStatusApiParameter = new LoggerParameter__mdt(DeveloperName = 'CallStatusApi', Value__c = 'true'); - LoggerParameter.setMockParameter(mockCallStatusApiParameter); - - String transactionId = '123-456-789-0'; - LogEntryEvent__e logEntryEvent = new LogEntryEvent__e( - Message__c = 'my message', - Timestamp__c = System.now(), - TransactionEntryNumber__c = 1, - TransactionId__c = transactionId - ); - + LoggerTestConfigurator.setMock(mockCallStatusApiParameter); + LogEntryEvent__e logEntryEvent = createLogEntryEvent(); Database.SaveResult saveResult; - try { - Test.startTest(); - - saveResult = EventBus.publish(logEntryEvent); - // Normally, you don't have to call Test.getEventBus().deliver() if you're also using Test.stopTest() - // But, in this case, there are 3 transactions happening (original test, async platform event, and async future method) - // So, call Test.getEventBus().deliver() and Test.stopTest() to make sure all transactions are completed before running asserts - Test.getEventBus().deliver(); - Test.stopTest(); + try { + // TODO Test.startTest() & stopTest() are still being used so that the callout in LogEntryEventHandler executes, + // but it be nice if this was instead mocked via LoggerMockDataStore (? or some other similar class?) + System.Test.startTest(); + saveResult = LoggerMockDataStore.getEventBus().publishRecord(logEntryEvent); + LoggerMockDataStore.getEventBus().deliver(new LogEntryEventHandler()); + System.Test.stopTest(); System.assert(false, 'Exception expected, this assert should not run'); } catch (Exception ex) { System.assertEquals(CalloutException.class.getName(), ex.getTypeName()); - String expectedErrorMessage = 'Callout failed for https://api.status.salesforce.com/v1/instances/'; System.assert(ex.getMessage().contains(expectedErrorMessage)); } System.assertEquals(true, saveResult.isSuccess(), saveResult.getErrors()); - + System.assertEquals( + 1, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed one time for AFTER_INSERT' + ); Log__c log = getLog(); System.assertEquals(1, log.LogEntries__r.size()); LogEntry__c logEntry = log.LogEntries__r.get(0); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); System.assertEquals(null, log.ApiReleaseNumber__c); System.assertEquals(null, log.ApiReleaseVersion__c); @@ -673,44 +573,33 @@ private class LogEntryEventHandler_Tests { validateLogEntryFields(logEntryEvent, logEntry); } - private class SuccessCalloutMock implements HttpCalloutMock { - public HttpResponse respond(HttpRequest request) { - LogEntryEventHandler.StatusApiResponse apiResponse = new LogEntryEventHandler.StatusApiResponse(); - apiResponse.releaseNumber = MOCK_RELEASE_NUMBER; - apiResponse.releaseVersion = MOCK_RELEASE_VERSION; - - HttpResponse response = new HttpResponse(); - response.setBody(JSON.serialize(apiResponse)); - response.setHeader('Content-Type', 'application/json'); - response.setStatusCode(200); - - return response; - } - } - - private class FailureCalloutMock implements HttpCalloutMock { - public HttpResponse respond(HttpRequest request) { - HttpResponse response = new HttpResponse(); - response.setStatusCode(400); - return response; - } + private static LogEntryEvent__e createLogEntryEvent() { + // The data builder class handles populating field values, but for some fields, + // certain values are expected (e.g., LoggedById__c should have a valid user ID), + // so this method handles any additional manipulation to the field values + LogEntryEvent__e logEntryEvent = (LogEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType) + .populateAllFields() + .getRecord(); + logEntryEvent.LoggedById__c = UserInfo.getUserId(); + logEntryEvent.LoggingLevel__c = LoggingLevel.INFO.name(); + logEntryEvent.LoggingLevelOrdinal__c = LoggingLevel.INFO.ordinal(); + logEntryEvent.UserLoggingLevel__c = LoggingLevel.INFO.name(); + logEntryEvent.UserLoggingLevelOrdinal__c = LoggingLevel.INFO.ordinal(); + logEntryEvent.ProfileId__c = UserInfo.getProfileId(); + logEntryEvent.TimestampString__c = String.valueOf(logEntryEvent.Timestamp__c.getTime()); + logEntryEvent.ParentLogTransactionId__c = null; + logEntryEvent.RecordId__c = UserInfo.getUserId(); + logEntryEvent.RecordCollectionSize__c = 1; + logEntryEvent.RecordCollectionType__c = 'Single'; + + return logEntryEvent; } - private static User getCurrentUser() { - return [ - SELECT - Id, - ProfileId, - Profile.Name, - Profile.UserLicenseId, - Profile.UserLicense.LicenseDefinitionKey, - Profile.UserLicense.Name, - Username, - UserRoleId, - UserRole.Name - FROM User - WHERE Id = :UserInfo.getUserId() - ]; + private static String createStatusApiResponseJson() { + LogEntryEventHandler.StatusApiResponse apiResponse = new LogEntryEventHandler.StatusApiResponse(); + apiResponse.releaseNumber = MOCK_RELEASE_NUMBER; + apiResponse.releaseVersion = MOCK_RELEASE_VERSION; + return JSON.serialize(apiResponse); } private static Log__c getLog() { @@ -761,9 +650,24 @@ private class LogEntryEventHandler_Tests { UserType__c, ( SELECT + ComponentType__c, + DatabaseResultCollectionSize__c, + DatabaseResultCollectionType__c, + DatabaseResultJson__c, + DatabaseResultType__c, EpochTimestamp__c, ExceptionStackTrace__c, ExceptionType__c, + HttpRequestBody__c, + HttpRequestBodyMasked__c, + HttpRequestCompressed__c, + HttpRequestEndpoint__c, + HttpRequestMethod__c, + HttpResponseBody__c, + HttpResponseBodyMasked__c, + HttpResponseHeaderKeys__c, + HttpResponseStatus__c, + HttpResponseStatusCode__c, LimitsAggregateQueriesMax__c, LimitsAggregateQueriesUsed__c, LimitsAsyncCallsMax__c, @@ -805,8 +709,11 @@ private class LogEntryEventHandler_Tests { Name, OriginType__c, OriginLocation__c, + RecordCollectionSize__c, + RecordCollectionType__c, RecordId__c, RecordJson__c, + RecordJsonMasked__c, RecordSObjectClassification__c, RecordSObjectType__c, RecordSObjectTypeNamespace__c, @@ -837,104 +744,263 @@ private class LogEntryEventHandler_Tests { Id logOwnerId = logEntryEvent.LoggedById__c == null ? UserInfo.getUserId() : logEntryEvent.LoggedById__c; - System.assertEquals(logEntryEvent.ApiVersion__c, log.ApiVersion__c); - System.assertEquals(logEntryEvent.Locale__c, log.Locale__c); - System.assertEquals(logEntryEvent.LoggedById__c, log.LoggedBy__c); - System.assertEquals(logEntryEvent.LoggedByUsername__c, log.LoggedByUsername__c); - System.assertEquals(logEntryEvent.LoggerVersionNumber__c, log.LoggerVersionNumber__c); - System.assertEquals(logEntryEvent.LoginDomain__c, log.LoginDomain__c); - System.assertEquals(logEntryEvent.LoginHistoryId__c, log.LoginHistoryId__c); - System.assertEquals(logEntryEvent.LoginType__c, log.LoginType__c); - System.assertEquals(logEntryEvent.LogoutUrl__c, log.LogoutUrl__c); - System.assertEquals(logEntryEvent.NetworkId__c, log.NetworkId__c); - System.assertEquals(logOwnerId, log.OwnerId); - System.assertEquals(logEntryEvent.ProfileId__c, log.ProfileId__c); - System.assertEquals(logEntryEvent.ProfileName__c, log.ProfileName__c); - System.assertEquals(logEntryEvent.Scenario__c, log.Scenario__c); - System.assertEquals(logEntryEvent.SessionId__c, log.SessionId__c); - System.assertEquals(logEntryEvent.SessionId__c, log.SessionId__c); - System.assertEquals(logEntryEvent.SessionSecurityLevel__c, log.SessionSecurityLevel__c); - System.assertEquals(logEntryEvent.SessionType__c, log.SessionType__c); - System.assertEquals(logEntryEvent.SourceIp__c, log.SourceIp__c); - System.assertEquals(logEntryEvent.SystemMode__c, log.SystemMode__c); - System.assertEquals(logEntryEvent.ThemeDisplayed__c, log.ThemeDisplayed__c); - System.assertEquals(logEntryEvent.TimeZoneId__c, log.TimeZoneId__c); - System.assertEquals(logEntryEvent.TimeZoneName__c, log.TimeZoneName__c); - System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c); - System.assertEquals(logEntryEvent.UserLoggingLevel__c, log.UserLoggingLevel__c); - System.assertEquals(logEntryEvent.UserLoggingLevelOrdinal__c, log.UserLoggingLevelOrdinal__c); - // System.assertEquals(currentUser.UserRoleId, log.UserRoleId__c); - // System.assertEquals(currentUser.UserRoleId == null ? null : currentUser.UserRole.Name, log.UserRoleName__c); - System.assertEquals(logEntryEvent.UserType__c, log.UserType__c); + System.assertEquals(logEntryEvent.ApiVersion__c, log.ApiVersion__c, 'log.ApiVersion__c was not properly set'); + System.assertEquals(logEntryEvent.Locale__c, log.Locale__c, 'log.Locale__c was not properly set'); + System.assertEquals(logEntryEvent.LoggedById__c, log.LoggedBy__c, 'log.LoggedBy__c was not properly set'); + System.assertEquals(logEntryEvent.LoggedByUsername__c, log.LoggedByUsername__c, 'log.LoggedByUsername__c was not properly set'); + System.assertEquals(logEntryEvent.LoggerVersionNumber__c, log.LoggerVersionNumber__c, 'log.LoggerVersionNumber__c was not properly set'); + System.assertEquals(logEntryEvent.LoginDomain__c, log.LoginDomain__c, 'log.LoginDomain__c was not properly set'); + System.assertEquals(logEntryEvent.LoginHistoryId__c, log.LoginHistoryId__c, 'log.LoginHistoryId__c was not properly set'); + System.assertEquals(logEntryEvent.LoginType__c, log.LoginType__c, 'log.LoginType__c was not properly set'); + System.assertEquals(logEntryEvent.LogoutUrl__c, log.LogoutUrl__c, 'log.LogoutUrl__c was not properly set'); + System.assertEquals(logEntryEvent.NetworkId__c, log.NetworkId__c, 'log.NetworkId__c was not properly set'); + System.assertEquals(logOwnerId, log.OwnerId, 'log.OwnerId was not properly set'); + System.assertEquals(logEntryEvent.ProfileId__c, log.ProfileId__c, 'log.ProfileId__c was not properly set'); + System.assertEquals(logEntryEvent.ProfileName__c, log.ProfileName__c, 'log.ProfileName__c was not properly set'); + System.assertEquals(logEntryEvent.Scenario__c, log.Scenario__c, 'log.Scenario__c was not properly set'); + System.assertEquals(logEntryEvent.SessionId__c, log.SessionId__c, 'log.SessionId__c was not properly set'); + System.assertEquals(logEntryEvent.SessionId__c, log.SessionId__c, 'log.SessionId__c was not properly set'); + System.assertEquals(logEntryEvent.SessionSecurityLevel__c, log.SessionSecurityLevel__c, 'log.SessionSecurityLevel__c was not properly set'); + System.assertEquals(logEntryEvent.SessionType__c, log.SessionType__c, 'log.SessionType__c was not properly set'); + System.assertEquals(logEntryEvent.SourceIp__c, log.SourceIp__c, 'log.SourceIp__c was not properly set'); + System.assertEquals(logEntryEvent.SystemMode__c, log.SystemMode__c, 'log.SystemMode__c was not properly set'); + System.assertEquals(logEntryEvent.ThemeDisplayed__c, log.ThemeDisplayed__c, 'log.ThemeDisplayed__c was not properly set'); + System.assertEquals(logEntryEvent.TimeZoneId__c, log.TimeZoneId__c, 'log.TimeZoneId__c was not properly set'); + System.assertEquals(logEntryEvent.TimeZoneName__c, log.TimeZoneName__c, 'log.TimeZoneName__c was not properly set'); + System.assertEquals(logEntryEvent.TransactionId__c, log.TransactionId__c, 'log.TransactionId__c was not properly set'); + System.assertEquals(logEntryEvent.UserLoggingLevel__c, log.UserLoggingLevel__c, 'log.UserLoggingLevel__c was not properly set'); + System.assertEquals(logEntryEvent.UserLoggingLevelOrdinal__c, log.UserLoggingLevelOrdinal__c, 'log.UserLoggingLevelOrdinal__c was not properly set'); + // System.assertEquals(currentUser.UserRoleId, log.UserRoleId__c, 'log.UserRoleId__c was not properly set'); + // System.assertEquals(currentUser.UserRoleId == null ? null : currentUser.UserRole.Name, log.UserRoleName__c, 'log.UserRoleName__c was not properly set'); + System.assertEquals(logEntryEvent.UserType__c, log.UserType__c, 'log.UserType__c was not properly set'); // Org fields - System.assertEquals(logEntryEvent.OrganizationDomainUrl__c, log.OrganizationDomainUrl__c); - System.assertEquals(logEntryEvent.OrganizationEnvironmentType__c, log.OrganizationEnvironmentType__c); - System.assertEquals(logEntryEvent.OrganizationId__c, log.OrganizationId__c); - System.assertEquals(logEntryEvent.OrganizationInstanceName__c, log.OrganizationInstanceName__c); - System.assertEquals(logEntryEvent.OrganizationName__c, log.OrganizationName__c); - System.assertEquals(logEntryEvent.OrganizationNamespacePrefix__c, log.OrganizationNamespacePrefix__c); - System.assertEquals(logEntryEvent.OrganizationType__c, log.OrganizationType__c); + System.assertEquals(logEntryEvent.OrganizationDomainUrl__c, log.OrganizationDomainUrl__c, 'log.OrganizationDomainUrl__c was not properly set'); + System.assertEquals( + logEntryEvent.OrganizationEnvironmentType__c, + log.OrganizationEnvironmentType__c, + 'log.OrganizationEnvironmentType__c was not properly set' + ); + System.assertEquals(logEntryEvent.OrganizationId__c, log.OrganizationId__c, 'log.OrganizationId__c was not properly set'); + System.assertEquals(logEntryEvent.OrganizationInstanceName__c, log.OrganizationInstanceName__c, 'log.OrganizationInstanceName__c was not properly set'); + System.assertEquals(logEntryEvent.OrganizationName__c, log.OrganizationName__c, 'log.OrganizationName__c was not properly set'); + System.assertEquals( + logEntryEvent.OrganizationNamespacePrefix__c, + log.OrganizationNamespacePrefix__c, + 'log.OrganizationNamespacePrefix__c was not properly set' + ); + System.assertEquals(logEntryEvent.OrganizationType__c, log.OrganizationType__c, 'log.OrganizationType__c was not properly set'); // Profile fields - System.assertEquals(logEntryEvent.UserLicenseDefinitionKey__c, log.UserLicenseDefinitionKey__c); - System.assertEquals(logEntryEvent.UserLicenseId__c, log.UserLicenseId__c); - System.assertEquals(logEntryEvent.UserLicenseName__c, log.UserLicenseName__c); + System.assertEquals(logEntryEvent.UserLicenseDefinitionKey__c, log.UserLicenseDefinitionKey__c, 'log.UserLicenseDefinitionKey__c was not properly set'); + System.assertEquals(logEntryEvent.UserLicenseId__c, log.UserLicenseId__c, 'log.UserLicenseId__c was not properly set'); + System.assertEquals(logEntryEvent.UserLicenseName__c, log.UserLicenseName__c, 'log.UserLicenseName__c was not properly set'); } private static void validateLogEntryFields(LogEntryEvent__e logEntryEvent, LogEntry__c logEntry) { - System.assertEquals(logEntryEvent.EpochTimestamp__c, logEntry.EpochTimestamp__c); - System.assertEquals(logEntryEvent.ExceptionStackTrace__c, logEntry.ExceptionStackTrace__c); - System.assertEquals(logEntryEvent.ExceptionType__c, logEntry.ExceptionType__c); - System.assertEquals(logEntryEvent.LimitsAggregateQueriesMax__c, logEntry.LimitsAggregateQueriesMax__c); - System.assertEquals(logEntryEvent.LimitsAggregateQueriesUsed__c, logEntry.LimitsAggregateQueriesUsed__c); - System.assertEquals(logEntryEvent.LimitsAsyncCallsMax__c, logEntry.LimitsAsyncCallsMax__c); - System.assertEquals(logEntryEvent.LimitsAsyncCallsUsed__c, logEntry.LimitsAsyncCallsUsed__c); - System.assertEquals(logEntryEvent.LimitsCalloutsMax__c, logEntry.LimitsCalloutsMax__c); - System.assertEquals(logEntryEvent.LimitsCalloutsUsed__c, logEntry.LimitsCalloutsUsed__c); - System.assertEquals(logEntryEvent.LimitsCpuTimeMax__c, logEntry.LimitsCpuTimeMax__c); - System.assertEquals(logEntryEvent.LimitsCpuTimeUsed__c, logEntry.LimitsCpuTimeUsed__c); - System.assertEquals(logEntryEvent.LimitsDmlRowsMax__c, logEntry.LimitsDmlRowsMax__c); - System.assertEquals(logEntryEvent.LimitsDmlRowsUsed__c, logEntry.LimitsDmlRowsUsed__c); - System.assertEquals(logEntryEvent.LimitsDmlStatementsMax__c, logEntry.LimitsDmlStatementsMax__c); - System.assertEquals(logEntryEvent.LimitsDmlStatementsUsed__c, logEntry.LimitsDmlStatementsUsed__c); - System.assertEquals(logEntryEvent.LimitsEmailInvocationsMax__c, logEntry.LimitsEmailInvocationsMax__c); - System.assertEquals(logEntryEvent.LimitsEmailInvocationsUsed__c, logEntry.LimitsEmailInvocationsUsed__c); - System.assertEquals(logEntryEvent.LimitsFutureCallsMax__c, logEntry.LimitsFutureCallsMax__c); - System.assertEquals(logEntryEvent.LimitsFutureCallsUsed__c, logEntry.LimitsFutureCallsUsed__c); - System.assertEquals(logEntryEvent.LimitsHeapSizeMax__c, logEntry.LimitsHeapSizeMax__c); - System.assertEquals(logEntryEvent.LimitsHeapSizeUsed__c, logEntry.LimitsHeapSizeUsed__c); - System.assertEquals(logEntryEvent.LimitsMobilePushApexCallsMax__c, logEntry.LimitsMobilePushApexCallsMax__c); - System.assertEquals(logEntryEvent.LimitsMobilePushApexCallsUsed__c, logEntry.LimitsMobilePushApexCallsUsed__c); - System.assertEquals(logEntryEvent.LimitsPublishImmediateDmlStatementsMax__c, logEntry.LimitsPublishImmediateDmlStatementsMax__c); - System.assertEquals(logEntryEvent.LimitsPublishImmediateDmlStatementsUsed__c, logEntry.LimitsPublishImmediateDmlStatementsUsed__c); - System.assertEquals(logEntryEvent.LimitsQueueableJobsMax__c, logEntry.LimitsQueueableJobsMax__c); - System.assertEquals(logEntryEvent.LimitsQueueableJobsUsed__c, logEntry.LimitsQueueableJobsUsed__c); - System.assertEquals(logEntryEvent.LimitsSoqlQueriesMax__c, logEntry.LimitsSoqlQueriesMax__c); - System.assertEquals(logEntryEvent.LimitsSoqlQueriesUsed__c, logEntry.LimitsSoqlQueriesUsed__c); - System.assertEquals(logEntryEvent.LimitsSoqlQueryLocatorRowsMax__c, logEntry.LimitsSoqlQueryLocatorRowsMax__c); - System.assertEquals(logEntryEvent.LimitsSoqlQueryLocatorRowsUsed__c, logEntry.LimitsSoqlQueryLocatorRowsUsed__c); - System.assertEquals(logEntryEvent.LimitsSoqlQueryRowsMax__c, logEntry.LimitsSoqlQueryRowsMax__c); - System.assertEquals(logEntryEvent.LimitsSoqlQueryRowsUsed__c, logEntry.LimitsSoqlQueryRowsUsed__c); - System.assertEquals(logEntryEvent.LimitsSoslSearchesUsed__c, logEntry.LimitsSoslSearchesUsed__c); - System.assertEquals(logEntryEvent.LimitsSoslSearchesMax__c, logEntry.LimitsSoslSearchesMax__c); - System.assertEquals(logEntryEvent.LoggingLevel__c, logEntry.LoggingLevel__c); - System.assertEquals(logEntryEvent.LoggingLevelOrdinal__c, logEntry.LoggingLevelOrdinal__c); - System.assertEquals(logEntryEvent.Message__c, logEntry.Message__c); - System.assertEquals(logEntryEvent.MessageTruncated__c, logEntry.MessageTruncated__c); - System.assertEquals(logEntry.Id, logEntry.Name); - System.assertEquals(logEntryEvent.OriginType__c, logEntry.OriginType__c); - System.assertEquals(logEntryEvent.OriginLocation__c, logEntry.OriginLocation__c); - System.assertEquals(logEntryEvent.RecordId__c, logEntry.RecordId__c); - System.assertEquals(logEntryEvent.RecordJson__c, logEntry.RecordJson__c); - System.assertEquals(logEntryEvent.RecordSObjectClassification__c, logEntry.RecordSObjectClassification__c); - System.assertEquals(logEntryEvent.RecordSObjectType__c, logEntry.RecordSObjectType__c); - System.assertEquals(logEntryEvent.RecordSObjectTypeNamespace__c, logEntry.RecordSObjectTypeNamespace__c); - System.assertEquals(logEntryEvent.StackTrace__c, logEntry.StackTrace__c); - System.assertEquals(logEntryEvent.Timestamp__c, logEntry.Timestamp__c); - System.assertEquals(logEntryEvent.TransactionEntryNumber__c, logEntry.TransactionEntryNumber__c); - System.assertEquals(logEntryEvent.TriggerIsExecuting__c, logEntry.TriggerIsExecuting__c); - System.assertEquals(logEntryEvent.TriggerOperationType__c, logEntry.TriggerOperationType__c); - System.assertEquals(logEntryEvent.TriggerSObjectType__c, logEntry.TriggerSObjectType__c); + System.assertEquals(logEntryEvent.ComponentType__c, logEntry.ComponentType__c, 'logEntry.ComponentType__c was not properly set'); + System.assertEquals( + logEntryEvent.DatabaseResultCollectionSize__c, + logEntry.DatabaseResultCollectionSize__c, + 'logEntry.DatabaseResultCollectionSize__c was not properly set' + ); + System.assertEquals( + logEntryEvent.DatabaseResultCollectionType__c, + logEntry.DatabaseResultCollectionType__c, + 'logEntry.DatabaseResultCollectionType__c was not properly set' + ); + System.assertEquals(logEntryEvent.DatabaseResultJson__c, logEntry.DatabaseResultJson__c, 'logEntry.DatabaseResultJson__c was not properly set'); + System.assertEquals(logEntryEvent.DatabaseResultType__c, logEntry.DatabaseResultType__c, 'logEntry.DatabaseResultType__c was not properly set'); + System.assertEquals(logEntryEvent.EpochTimestamp__c, logEntry.EpochTimestamp__c, 'logEntry.EpochTimestamp__c was not properly set'); + System.assertEquals(logEntryEvent.ExceptionStackTrace__c, logEntry.ExceptionStackTrace__c, 'logEntry.ExceptionStackTrace__c was not properly set'); + System.assertEquals(logEntryEvent.ExceptionType__c, logEntry.ExceptionType__c, 'logEntry.ExceptionType__c was not properly set'); + System.assertEquals(logEntryEvent.HttpRequestBody__c, logEntry.HttpRequestBody__c, 'logEntry.HttpRequestBody__c was not properly set'); + System.assertEquals( + logEntryEvent.HttpRequestBodyMasked__c, + logEntry.HttpRequestBodyMasked__c, + 'logEntry.HttpRequestBodyMasked__c was not properly set' + ); + System.assertEquals( + logEntryEvent.HttpRequestCompressed__c, + logEntry.HttpRequestCompressed__c, + 'logEntry.HttpRequestCompressed__c was not properly set' + ); + System.assertEquals(logEntryEvent.HttpRequestEndpoint__c, logEntry.HttpRequestEndpoint__c, 'logEntry.HttpRequestEndpoint__c was not properly set'); + System.assertEquals(logEntryEvent.HttpRequestMethod__c, logEntry.HttpRequestMethod__c, 'logEntry.HttpRequestMethod__c was not properly set'); + System.assertEquals(logEntryEvent.HttpResponseBody__c, logEntry.HttpResponseBody__c, 'logEntry.HttpResponseBody__c was not properly set'); + System.assertEquals( + logEntryEvent.HttpResponseBodyMasked__c, + logEntry.HttpResponseBodyMasked__c, + 'logEntry.HttpResponseBodyMasked__c was not properly set' + ); + System.assertEquals( + logEntryEvent.HttpResponseHeaderKeys__c, + logEntry.HttpResponseHeaderKeys__c, + 'logEntry.HttpResponseHeaderKeys__c was not properly set' + ); + System.assertEquals(logEntryEvent.HttpResponseStatus__c, logEntry.HttpResponseStatus__c, 'logEntry.HttpResponseStatus__c was not properly set'); + System.assertEquals( + logEntryEvent.HttpResponseStatusCode__c, + logEntry.HttpResponseStatusCode__c, + 'logEntry.HttpResponseStatusCode__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsAggregateQueriesMax__c, + logEntry.LimitsAggregateQueriesMax__c, + 'logEntry.LimitsAggregateQueriesMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsAggregateQueriesUsed__c, + logEntry.LimitsAggregateQueriesUsed__c, + 'logEntry.LimitsAggregateQueriesUsed__c was not properly set' + ); + System.assertEquals(logEntryEvent.LimitsAsyncCallsMax__c, logEntry.LimitsAsyncCallsMax__c, 'logEntry.LimitsAsyncCallsMax__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsAsyncCallsUsed__c, logEntry.LimitsAsyncCallsUsed__c, 'logEntry.LimitsAsyncCallsUsed__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsCalloutsMax__c, logEntry.LimitsCalloutsMax__c, 'logEntry.LimitsCalloutsMax__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsCalloutsUsed__c, logEntry.LimitsCalloutsUsed__c, 'logEntry.LimitsCalloutsUsed__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsCpuTimeMax__c, logEntry.LimitsCpuTimeMax__c, 'logEntry.LimitsCpuTimeMax__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsCpuTimeUsed__c, logEntry.LimitsCpuTimeUsed__c, 'logEntry.LimitsCpuTimeUsed__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsDmlRowsMax__c, logEntry.LimitsDmlRowsMax__c, 'logEntry.LimitsDmlRowsMax__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsDmlRowsUsed__c, logEntry.LimitsDmlRowsUsed__c, 'logEntry.LimitsDmlRowsUsed__c was not properly set'); + System.assertEquals( + logEntryEvent.LimitsDmlStatementsMax__c, + logEntry.LimitsDmlStatementsMax__c, + 'logEntry.LimitsDmlStatementsMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsDmlStatementsUsed__c, + logEntry.LimitsDmlStatementsUsed__c, + 'logEntry.LimitsDmlStatementsUsed__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsEmailInvocationsMax__c, + logEntry.LimitsEmailInvocationsMax__c, + 'logEntry.LimitsEmailInvocationsMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsEmailInvocationsUsed__c, + logEntry.LimitsEmailInvocationsUsed__c, + 'logEntry.LimitsEmailInvocationsUsed__c was not properly set' + ); + System.assertEquals(logEntryEvent.LimitsFutureCallsMax__c, logEntry.LimitsFutureCallsMax__c, 'logEntry.LimitsFutureCallsMax__c was not properly set'); + System.assertEquals( + logEntryEvent.LimitsFutureCallsUsed__c, + logEntry.LimitsFutureCallsUsed__c, + 'logEntry.LimitsFutureCallsUsed__c was not properly set' + ); + System.assertEquals(logEntryEvent.LimitsHeapSizeMax__c, logEntry.LimitsHeapSizeMax__c, 'logEntry.LimitsHeapSizeMax__c was not properly set'); + System.assertEquals(logEntryEvent.LimitsHeapSizeUsed__c, logEntry.LimitsHeapSizeUsed__c, 'logEntry.LimitsHeapSizeUsed__c was not properly set'); + System.assertEquals( + logEntryEvent.LimitsMobilePushApexCallsMax__c, + logEntry.LimitsMobilePushApexCallsMax__c, + 'logEntry.LimitsMobilePushApexCallsMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsMobilePushApexCallsUsed__c, + logEntry.LimitsMobilePushApexCallsUsed__c, + 'logEntry.LimitsMobilePushApexCallsUsed__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsPublishImmediateDmlStatementsMax__c, + logEntry.LimitsPublishImmediateDmlStatementsMax__c, + 'logEntry.LimitsPublishImmediateDmlStatementsMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsPublishImmediateDmlStatementsUsed__c, + logEntry.LimitsPublishImmediateDmlStatementsUsed__c, + 'logEntry.LimitsPublishImmediateDmlStatementsUsed__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsQueueableJobsMax__c, + logEntry.LimitsQueueableJobsMax__c, + 'logEntry.LimitsQueueableJobsMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsQueueableJobsUsed__c, + logEntry.LimitsQueueableJobsUsed__c, + 'logEntry.LimitsQueueableJobsUsed__c was not properly set' + ); + System.assertEquals(logEntryEvent.LimitsSoqlQueriesMax__c, logEntry.LimitsSoqlQueriesMax__c, 'logEntry.LimitsSoqlQueriesMax__c was not properly set'); + System.assertEquals( + logEntryEvent.LimitsSoqlQueriesUsed__c, + logEntry.LimitsSoqlQueriesUsed__c, + 'logEntry.LimitsSoqlQueriesUsed__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsSoqlQueryLocatorRowsMax__c, + logEntry.LimitsSoqlQueryLocatorRowsMax__c, + 'logEntry.LimitsSoqlQueryLocatorRowsMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsSoqlQueryLocatorRowsUsed__c, + logEntry.LimitsSoqlQueryLocatorRowsUsed__c, + 'logEntry.LimitsSoqlQueryLocatorRowsUsed__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsSoqlQueryRowsMax__c, + logEntry.LimitsSoqlQueryRowsMax__c, + 'logEntry.LimitsSoqlQueryRowsMax__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsSoqlQueryRowsUsed__c, + logEntry.LimitsSoqlQueryRowsUsed__c, + 'logEntry.LimitsSoqlQueryRowsUsed__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsSoslSearchesUsed__c, + logEntry.LimitsSoslSearchesUsed__c, + 'logEntry.LimitsSoslSearchesUsed__c was not properly set' + ); + System.assertEquals( + logEntryEvent.LimitsSoslSearchesMax__c, + logEntry.LimitsSoslSearchesMax__c, + 'logEntry.LimitsSoslSearchesMax__c was not properly set' + ); + System.assertEquals(logEntryEvent.LoggingLevel__c, logEntry.LoggingLevel__c, 'logEntry.LoggingLevel__c was not properly set'); + System.assertEquals(logEntryEvent.LoggingLevelOrdinal__c, logEntry.LoggingLevelOrdinal__c, 'logEntry.LoggingLevelOrdinal__c was not properly set'); + System.assertEquals(logEntryEvent.Message__c, logEntry.Message__c, 'logEntry.Message__c was not properly set'); + System.assertEquals(logEntryEvent.MessageTruncated__c, logEntry.MessageTruncated__c, 'logEntry.MessageTruncated__c was not properly set'); + System.assertEquals(logEntry.Id, logEntry.Name, 'logEntry.Name was not properly set'); + System.assertEquals(logEntryEvent.OriginType__c, logEntry.OriginType__c, 'logEntry.OriginType__c was not properly set'); + System.assertEquals(logEntryEvent.OriginLocation__c, logEntry.OriginLocation__c, 'logEntry.OriginLocation__c was not properly set'); + System.assertEquals(logEntryEvent.RecordCollectionSize__c, logEntry.RecordCollectionSize__c, 'logEntry.RecordCollectionSize__c was not properly set'); + System.assertEquals(logEntryEvent.RecordCollectionType__c, logEntry.RecordCollectionType__c, 'logEntry.RecordCollectionType__c was not properly set'); + System.assertEquals(logEntryEvent.RecordId__c, logEntry.RecordId__c, 'logEntry.RecordId__c was not properly set'); + System.assertEquals(logEntryEvent.RecordJson__c, logEntry.RecordJson__c, 'logEntry.RecordJson__c was not properly set'); + System.assertEquals(logEntryEvent.RecordJsonMasked__c, logEntry.RecordJsonMasked__c, 'logEntry.RecordJsonMasked__c was not properly set'); + System.assertEquals( + logEntryEvent.RecordSObjectClassification__c, + logEntry.RecordSObjectClassification__c, + 'logEntry.RecordSObjectClassification__c was not properly set' + ); + System.assertEquals(logEntryEvent.RecordSObjectType__c, logEntry.RecordSObjectType__c, 'logEntry.RecordSObjectType__c was not properly set'); + System.assertEquals( + logEntryEvent.RecordSObjectTypeNamespace__c, + logEntry.RecordSObjectTypeNamespace__c, + 'logEntry.RecordSObjectTypeNamespace__c was not properly set' + ); + System.assertEquals(logEntryEvent.RecordId__c, logEntry.RecordId__c, 'logEntry.RecordId__c was not properly set'); + System.assertEquals(logEntryEvent.RecordJson__c, logEntry.RecordJson__c, 'logEntry.RecordJson__c was not properly set'); + System.assertEquals( + logEntryEvent.RecordSObjectClassification__c, + logEntry.RecordSObjectClassification__c, + 'logEntry.RecordSObjectClassification__c was not properly set' + ); + System.assertEquals(logEntryEvent.RecordSObjectType__c, logEntry.RecordSObjectType__c, 'logEntry.RecordSObjectType__c was not properly set'); + System.assertEquals( + logEntryEvent.RecordSObjectTypeNamespace__c, + logEntry.RecordSObjectTypeNamespace__c, + 'logEntry.RecordSObjectTypeNamespace__c was not properly set' + ); + System.assertEquals(logEntryEvent.StackTrace__c, logEntry.StackTrace__c, 'logEntry.StackTrace__c was not properly set'); + System.assertEquals(logEntryEvent.Timestamp__c, logEntry.Timestamp__c, 'logEntry.Timestamp__c was not properly set'); + System.assertEquals( + logEntryEvent.TransactionEntryNumber__c, + logEntry.TransactionEntryNumber__c, + 'logEntry.TransactionEntryNumber__c was not properly set' + ); + System.assertEquals(logEntryEvent.TriggerIsExecuting__c, logEntry.TriggerIsExecuting__c, 'logEntry.TriggerIsExecuting__c was not properly set'); + System.assertEquals(logEntryEvent.TriggerOperationType__c, logEntry.TriggerOperationType__c, 'logEntry.TriggerOperationType__c was not properly set'); + System.assertEquals(logEntryEvent.TriggerSObjectType__c, logEntry.TriggerSObjectType__c, 'logEntry.TriggerSObjectType__c was not properly set'); } } diff --git a/nebula-logger/core/tests/log-management/classes/LogEntryFieldSetPicklist_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogEntryFieldSetPicklist_Tests.cls index de4fefd1f..9fb7bbf8b 100644 --- a/nebula-logger/core/tests/log-management/classes/LogEntryFieldSetPicklist_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogEntryFieldSetPicklist_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LogEntryFieldSetPicklist_Tests { @IsTest static void it_should_return_null_for_default_value() { diff --git a/nebula-logger/core/tests/log-management/classes/LogEntryHandler_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogEntryHandler_Tests.cls index a73e05666..182082e55 100644 --- a/nebula-logger/core/tests/log-management/classes/LogEntryHandler_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogEntryHandler_Tests.cls @@ -4,57 +4,46 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LogEntryHandler_Tests { - public class LogEntryPluginTest extends LoggerSObjectHandlerPlugin { - public override void execute( - TriggerOperation triggerOperationType, - List triggerNew, - Map triggerNewMap, - List triggerOld, - Map triggerOldMap - ) { - if (triggerOperationType == TriggerOperation.BEFORE_INSERT) { - for (LogEntry__c logEntry : (List) triggerNew) { - // The specific field changed doesn't really matter - we just want to ensure that whatever - // logic implemented in the instance of LoggerSObjectHandlerPlugin is executed - logEntry.Message__c = 'Some String'; - } - } - } - } - - private static String getNamespacePrefix() { - String className = LogEntryHandler_Tests.class.getName(); - String namespacePrefix = className.contains('.') ? className.substringBefore('.') : ''; - - return namespacePrefix; - } - @TestSetup static void setupData() { - Log__c log = new Log__c(TransactionId__c = '1234'); + LoggerSObjectHandler.shouldExecute(false); + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); insert log; Test.setCreatedDate(log.Id, System.now().addDays(-8)); } @IsTest static void it_should_return_the_logEntry_sobjectType() { - Test.startTest(); System.assertEquals(Schema.LogEntry__c.SObjectType, new LogEntryHandler().getSObjectType()); - Test.stopTest(); } @IsTest - static void it_should_save_log_entry_without_related_record_id() { + static void it_should_not_run_when_disabled_via_configuration() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntry__c.SObjectType).IsEnabled__c = false; Log__c log = [SELECT Id FROM Log__c LIMIT 1]; + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals(0, LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), 'Handler class should not have executed'); + } + + @IsTest + static void it_should_save_log_entry_without_related_record_id() { + Log__c log = [SELECT Id FROM Log__c LIMIT 1]; LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, RecordId__c = null); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, RecordId__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assertEquals(null, logEntry.RecordId__c); } @@ -63,13 +52,15 @@ private class LogEntryHandler_Tests { static void it_should_populate_related_record_fields_on_log_entry_with_related_user_record_id() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; User currentUser = [SELECT Id, Username FROM User WHERE Id = :UserInfo.getUserId()]; - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, RecordId__c = currentUser.Id); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, RecordId__c, RecordName__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assertEquals(currentUser.Id, logEntry.RecordId__c); System.assertEquals(currentUser.Username, logEntry.RecordName__c); @@ -79,13 +70,15 @@ private class LogEntryHandler_Tests { static void it_should_populate_related_record_fields_on_log_entry_with_related_profile_record_id() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; Profile currentProfile = [SELECT Id, Name, Description FROM Profile WHERE Id = :UserInfo.getProfileId()]; - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, RecordId__c = currentProfile.Id); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, RecordId__c, RecordName__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assertEquals(currentProfile.Id, logEntry.RecordId__c); System.assertEquals(currentProfile.Name, logEntry.RecordName__c); @@ -94,14 +87,17 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasRecordJson_to_true_when_populated() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - String recordJson = '{}'; LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, RecordJson__c = recordJson); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, HasRecordJson__c, RecordJson__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(logEntry.HasRecordJson__c); System.assertEquals(recordJson, logEntry.RecordJson__c); @@ -110,20 +106,21 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasRecordJson_to_true_when_updated() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, RecordJson__c = null); - insert logEntry; + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + LoggerDataStore.getDatabase().insertRecord(logEntry); logEntry = [SELECT Id, RecordJson__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assertEquals(null, logEntry.RecordJson__c); - - Test.startTest(); - String recordJson = '{}'; logEntry.RecordJson__c = recordJson; - update logEntry; - Test.stopTest(); + update logEntry; + System.assertEquals( + 4, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed four times - two times for BEFORE_INSERT/AFTER_INSERT' + ' and two more times for BEFORE_UPDATE/AFTER_UPDATE' + ); logEntry = [SELECT Id, HasRecordJson__c, RecordJson__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(logEntry.HasRecordJson__c); System.assertEquals(recordJson, logEntry.RecordJson__c); @@ -132,13 +129,16 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasExceptionStackTrace_to_false_when_null() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, ExceptionStackTrace__c = null); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, HasExceptionStackTrace__c, ExceptionStackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(!logEntry.HasExceptionStackTrace__c); System.assertEquals(null, logEntry.ExceptionStackTrace__c); @@ -147,14 +147,17 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasExceptionStackTrace_to_true_when_populated() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - String stackTrace = 'something'; LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, ExceptionStackTrace__c = stackTrace); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, HasExceptionStackTrace__c, ExceptionStackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(logEntry.HasExceptionStackTrace__c); System.assertEquals(stackTrace, logEntry.ExceptionStackTrace__c); @@ -163,20 +166,21 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasExceptionStackTrace_to_true_when_updated() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, ExceptionStackTrace__c = null); - insert logEntry; + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + LoggerDataStore.getDatabase().insertRecord(logEntry); logEntry = [SELECT Id, ExceptionStackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assertEquals(null, logEntry.ExceptionStackTrace__c); - Test.startTest(); - String stackTrace = 'something'; logEntry.ExceptionStackTrace__c = stackTrace; update logEntry; - Test.stopTest(); - + System.assertEquals( + 4, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed four times - two times for BEFORE_INSERT/AFTER_INSERT' + ' and two more times for BEFORE_UPDATE/AFTER_UPDATE' + ); logEntry = [SELECT Id, HasExceptionStackTrace__c, ExceptionStackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(logEntry.HasExceptionStackTrace__c); System.assertEquals(stackTrace, logEntry.ExceptionStackTrace__c); @@ -185,13 +189,16 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasStackTrace_to_false_when_null() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, StackTrace__c = null); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, HasStackTrace__c, StackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(!logEntry.HasStackTrace__c); System.assertEquals(null, logEntry.StackTrace__c); @@ -200,14 +207,17 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasStackTrace_to_true_when_populated() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - String stackTrace = 'something'; LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, StackTrace__c = stackTrace); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); - Test.startTest(); - insert logEntry; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(logEntry); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [SELECT Id, HasStackTrace__c, StackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(logEntry.HasStackTrace__c); System.assertEquals(stackTrace, logEntry.StackTrace__c); @@ -216,20 +226,21 @@ private class LogEntryHandler_Tests { @IsTest static void it_should_set_hasStackTrace_to_true_when_updated() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, StackTrace__c = null); - insert logEntry; + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + LoggerDataStore.getDatabase().insertRecord(logEntry); logEntry = [SELECT Id, StackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assertEquals(null, logEntry.StackTrace__c); - Test.startTest(); - String stackTrace = 'something'; logEntry.StackTrace__c = stackTrace; update logEntry; - Test.stopTest(); - + System.assertEquals( + 4, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed four times - two times for BEFORE_INSERT/AFTER_INSERT' + ' and two more times for BEFORE_UPDATE/AFTER_UPDATE' + ); logEntry = [SELECT Id, HasStackTrace__c, StackTrace__c FROM LogEntry__c WHERE Id = :logEntry.Id]; System.assert(logEntry.HasStackTrace__c); System.assertEquals(stackTrace, logEntry.StackTrace__c); @@ -239,7 +250,15 @@ private class LogEntryHandler_Tests { static void it_should_set_skip_setting_apex_class_details_when_origin_location_is_null() { Log__c log = [SELECT Id FROM Log__c LIMIT 1]; LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, OriginLocation__c = null, OriginType__c = 'Apex'); - insert logEntry; + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + + LoggerDataStore.getDatabase().insertRecord(logEntry); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [ SELECT Id, @@ -276,10 +295,17 @@ private class LogEntryHandler_Tests { FROM ApexClass WHERE NamespacePrefix = :getNamespacePrefix() AND Name = :exampleTopLevelClassName ]; - Log__c log = [SELECT Id FROM Log__c LIMIT 1]; LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, OriginLocation__c = exampleTopLevelClassMethodName, OriginType__c = 'Apex'); - insert logEntry; + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + + LoggerDataStore.getDatabase().insertRecord(logEntry); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [ SELECT Id, @@ -315,10 +341,17 @@ private class LogEntryHandler_Tests { FROM ApexClass WHERE NamespacePrefix = :getNamespacePrefix() AND Name = :exampleTopLevelClassName ]; - Log__c log = [SELECT Id FROM Log__c LIMIT 1]; LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, OriginLocation__c = exampleInnerClassMethodName, OriginType__c = 'Apex'); - insert logEntry; + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + + LoggerDataStore.getDatabase().insertRecord(logEntry); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntry__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntry = [ SELECT Id, @@ -343,24 +376,10 @@ private class LogEntryHandler_Tests { System.assertEquals(methodName, logEntry.ApexMethodName__c); } - @IsTest - static void it_should_run_apex_plugin_when_configured() { - String expectedMessage = 'Some String'; - - Log__c log = [SELECT Id FROM Log__c LIMIT 1]; - - Test.startTest(); - - // Use the mock configurations - LoggerPlugin__mdt plugin = new LoggerPlugin__mdt(PluginType__c = 'Apex', PluginApiName__c = LogEntryPluginTest.class.getName()); - LoggerSObjectHandler.setMockPlugin(Schema.LogEntry__c.SObjectType, plugin); - - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, Message__c = 'qwerty'); - insert logEntry; - - logEntry = [SELECT Id, Message__c FROM LogEntry__c WHERE Id = :logEntry.Id]; - System.assertEquals(expectedMessage, logEntry.Message__c); + private static String getNamespacePrefix() { + String className = LogEntryHandler_Tests.class.getName(); + String namespacePrefix = className.contains('.') ? className.substringBefore('.') : ''; - Test.stopTest(); + return namespacePrefix; } } diff --git a/nebula-logger/core/tests/log-management/classes/LogEntryTagHandler_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogEntryTagHandler_Tests.cls index 5a22f06b6..b9bdc7aea 100644 --- a/nebula-logger/core/tests/log-management/classes/LogEntryTagHandler_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogEntryTagHandler_Tests.cls @@ -4,37 +4,62 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LogEntryTagHandler_Tests { @TestSetup static void setupData() { - Log__c log = new Log__c(TransactionId__c = '1234'); + LoggerSObjectHandler.shouldExecute(false); + + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); insert log; List logEntries = new List(); for (Integer i = 0; i < 3; i++) { - logEntries.add(new LogEntry__c(Log__c = log.Id, Message__c = 'log entry #' + i)); + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, TransactionEntryNumber__c = i + 1); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + logEntries.add(logEntry); } insert logEntries; - insert new LoggerTag__c(Name = 'Some tag'); + insert LoggerMockDataCreator.createDataBuilder(Schema.LoggerTag__c.SObjectType).populateRequiredFields().getRecord(); } @IsTest - static void it_should_return_the_logEntry_sobjectType() { - Test.startTest(); + static void it_should_return_the_logEntryTag_sobjectType() { System.assertEquals(Schema.LogEntryTag__c.SObjectType, new LogEntryTagHandler().getSObjectType()); - Test.stopTest(); } @IsTest - static void it_should_set_unique_id_on_insert() { + static void it_should_not_run_when_disabled_via_configuration() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(false); LogEntry__c logEntry = [SELECT Id FROM LogEntry__c LIMIT 1]; LoggerTag__c tag = [SELECT Id FROM LoggerTag__c LIMIT 1]; + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(true); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LogEntryTag__c.SObjectType).IsEnabled__c = false; + LogEntryTag__c logEntryTag = new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id); + + LoggerDataStore.getDatabase().insertRecord(logEntryTag); + + System.assertEquals( + 0, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryTag__c.SObjectType).size(), + 'Handler class should not have executed' + ); + } + @IsTest + static void it_should_set_unique_id_on_insert() { + LogEntry__c logEntry = [SELECT Id FROM LogEntry__c LIMIT 1]; + LoggerTag__c tag = [SELECT Id FROM LoggerTag__c LIMIT 1]; LogEntryTag__c logEntryTag = new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id); - insert logEntryTag; + LoggerDataStore.getDatabase().insertRecord(logEntryTag); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryTag__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); logEntryTag = [SELECT Id, LogEntry__c, Tag__c, UniqueId__c FROM LogEntryTag__c WHERE Id = :logEntryTag.Id]; System.assertEquals(logEntry.Id + '' + tag.Id, logEntryTag.UniqueId__c); } @@ -43,12 +68,19 @@ private class LogEntryTagHandler_Tests { static void it_should_set_unique_id_on_update() { LogEntry__c logEntry = [SELECT Id FROM LogEntry__c LIMIT 1]; LoggerTag__c tag = [SELECT Id FROM LoggerTag__c LIMIT 1]; - LogEntryTag__c logEntryTag = new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id); - insert logEntryTag; + LoggerDataStore.getDatabase().insertRecord(logEntryTag); + logEntryTag = [SELECT Id, LogEntry__c, Tag__c, UniqueId__c FROM LogEntryTag__c WHERE Id = :logEntryTag.Id]; + System.assertEquals(logEntry.Id + '' + tag.Id, logEntryTag.UniqueId__c); logEntryTag.UniqueId__c = 'something else'; + update logEntryTag; + System.assertEquals( + 4, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryTag__c.SObjectType).size(), + 'Handler class should have executed four times - two times for BEFORE_INSERT/AFTER_INSERT' + ' and two more times for BEFORE_UPDATE/AFTER_UPDATE' + ); logEntryTag = [SELECT Id, LogEntry__c, Tag__c, UniqueId__c FROM LogEntryTag__c WHERE Id = :logEntryTag.Id]; System.assertEquals(logEntry.Id + '' + tag.Id, logEntryTag.UniqueId__c); } @@ -57,17 +89,26 @@ private class LogEntryTagHandler_Tests { static void it_should_not_insert_duplicate_tag() { LogEntry__c logEntry = [SELECT Id FROM LogEntry__c LIMIT 1]; LoggerTag__c tag = [SELECT Id FROM LoggerTag__c LIMIT 1]; - LogEntryTag__c logEntryTag = new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id); - insert logEntryTag; - + LoggerDataStore.getDatabase().insertRecord(logEntryTag); LogEntryTag__c duplicateLogEntryTag = new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id); + Exception thrownException; + try { insert duplicateLogEntryTag; System.assert(false, 'Exception expected on previous line'); } catch (Exception ex) { - String expectedDuplicateError = 'DUPLICATE_VALUE'; - System.assert(ex.getMessage().contains(expectedDuplicateError), ex.getMessage()); + thrownException = ex; } + + System.assertEquals( + 3, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryTag__c.SObjectType).size(), + 'Handler class should have executed three times - once for BEFORE_INSERT and once for AFTER_INSERT for the first record,' + + ' and once for BEFORE_INSERT on the errored duplicate' + ); + System.assertNotEquals(null, thrownException, 'An exception should have been thrown'); + String expectedDuplicateError = 'DUPLICATE_VALUE'; + System.assert(thrownException.getMessage().contains(expectedDuplicateError), thrownException.getMessage()); } } diff --git a/nebula-logger/core/tests/log-management/classes/LogHandler_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogHandler_Tests.cls index 1a8994bc8..99b9a3fce 100644 --- a/nebula-logger/core/tests/log-management/classes/LogHandler_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogHandler_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=false) private class LogHandler_Tests { private static final String HIGH_PRIORITY = Schema.Log__c.Priority__c.getDescribe().getPicklistValues().get(0).getValue(); private static final String MEDIUM_PRIORITY = Schema.Log__c.Priority__c.getDescribe().getPicklistValues().get(1).getValue(); @@ -13,45 +13,32 @@ private class LogHandler_Tests { private static final String FIRST_STATUS = Schema.Log__c.Status__c.getDescribe().getPicklistValues().get(0).getValue(); private static final String SECOND_STATUS = Schema.Log__c.Status__c.getDescribe().getPicklistValues().get(1).getValue(); - public class LogPluginTest extends LoggerSObjectHandlerPlugin { - public override void execute( - TriggerOperation triggerOperationType, - List triggerNew, - Map triggerNewMap, - List triggerOld, - Map triggerOldMap - ) { - if (triggerOperationType == TriggerOperation.BEFORE_INSERT) { - for (Log__c log : (List) triggerNew) { - // The specific field changed doesn't really matter - we just want to ensure that whatever - // logic implemented in the instance of LoggerSObjectHandlerPlugin is executed - log.ProfileName__c = 'Some String'; - } - } - } + @IsTest + static void it_should_return_the_log_sobjectType() { + System.assertEquals(Schema.Log__c.SObjectType, new LogHandler().getSObjectType()); } - @TestSetup - static void setupData() { - Map logStatusByName = new Map(); - LogStatus__mdt openStatus = new LogStatus__mdt(IsActive__c = true, MasterLabel = FIRST_STATUS, IsClosed__c = false, IsResolved__c = false); - logStatusByName.put(openStatus.MasterLabel, openStatus); - - LogStatus__mdt closedStatus = new LogStatus__mdt(IsActive__c = true, MasterLabel = SECOND_STATUS, IsClosed__c = true, IsResolved__c = true); - logStatusByName.put(closedStatus.MasterLabel, closedStatus); + @IsTest + static void it_should_not_run_when_disabled_via_configuration() { + setupConfigurations(); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(false); + Log__c log = new Log__c( + ClosedBy__c = UserInfo.getUserId(), + ClosedDate__c = System.now(), + IsClosed__c = true, + IsResolved__c = true, + Status__c = FIRST_STATUS, + TransactionId__c = '1234' + ); - LogHandler.LOG_STATUS_NAME_TO_STATUS.putAll(logStatusByName); - } + LoggerDataStore.getDatabase().insertRecord(log); - @IsTest - static void it_should_return_the_log_sobjectType() { - Test.startTest(); - System.assertEquals(Schema.Log__c.SObjectType, new LogHandler().getSObjectType()); - Test.stopTest(); + System.assertEquals(0, LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), 'Handler class should not have executed'); } @IsTest static void it_should_clear_closed_status_fields_when_open() { + setupConfigurations(); Log__c log = new Log__c( ClosedBy__c = UserInfo.getUserId(), ClosedDate__c = System.now(), @@ -61,10 +48,13 @@ private class LogHandler_Tests { TransactionId__c = '1234' ); - Test.startTest(); - insert log; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(log); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, ClosedBy__c, ClosedDate__c, IsClosed__c, IsResolved__c, Status__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(null, log.ClosedBy__c); System.assertEquals(null, log.ClosedDate__c); @@ -74,6 +64,7 @@ private class LogHandler_Tests { @IsTest static void it_should_set_closed_status_fields_when_closed() { + setupConfigurations(); Log__c log = new Log__c( ClosedBy__c = null, ClosedDate__c = null, @@ -83,10 +74,13 @@ private class LogHandler_Tests { TransactionId__c = '1234' ); - Test.startTest(); - insert log; - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(log); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, ClosedBy__c, ClosedDate__c, IsClosed__c, IsResolved__c, Status__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(UserInfo.getUserId(), log.ClosedBy__c); System.assertEquals(System.today(), log.ClosedDate__c.date()); @@ -95,194 +89,276 @@ private class LogHandler_Tests { @IsTest static void it_should_set_owner_when_default_configured_with_user_id() { + setupConfigurations(); User currentUser = new User(Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); - User expectedLogOwnerUser = LoggerTestUtils.createUser(); + User expectedLogOwnerUser = LoggerMockDataCreator.createUser(); insert expectedLogOwnerUser; LoggerSettings__c currentUserSettings = Logger.getUserSettings(currentUser); currentUserSettings.DefaultLogOwner__c = expectedLogOwnerUser.Id; insert currentUserSettings; - Log__c log = new Log__c(LoggedBy__c = currentUser.Id, OwnerId = currentUser.Id, TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, OwnerId FROM Log__c WHERE Id = :log.Id]; System.assertEquals(expectedLogOwnerUser.Id, log.OwnerId, log); } @IsTest static void it_should_set_owner_when_default_configured_with_username() { + setupConfigurations(); User currentUser = new User(Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); - User expectedLogOwnerUser = LoggerTestUtils.createUser(); + User expectedLogOwnerUser = LoggerMockDataCreator.createUser(); insert expectedLogOwnerUser; LoggerSettings__c currentUserSettings = Logger.getUserSettings(currentUser); currentUserSettings.DefaultLogOwner__c = expectedLogOwnerUser.Username; insert currentUserSettings; - Log__c log = new Log__c(LoggedBy__c = currentUser.Id, OwnerId = currentUser.Id, TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, OwnerId FROM Log__c WHERE Id = :log.Id]; System.assertEquals(expectedLogOwnerUser.Id, log.OwnerId, log); } @IsTest static void it_should_set_owner_when_default_configured_with_queue_id() { + setupConfigurations(); User currentUser = new User(Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); - Group expectedLogOwnerQueue = LoggerTestUtils.insertQueue(Schema.Log__c.SObjectType); + Group expectedLogOwnerQueue = LoggerMockDataCreator.insertQueue('Some_Log_Queue', Schema.Log__c.SObjectType); LoggerSettings__c currentUserSettings = Logger.getUserSettings(currentUser); currentUserSettings.DefaultLogOwner__c = expectedLogOwnerQueue.Id; insert currentUserSettings; - Log__c log = new Log__c(LoggedBy__c = currentUser.Id, OwnerId = currentUser.Id, TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, OwnerId FROM Log__c WHERE Id = :log.Id]; System.assertEquals(expectedLogOwnerQueue.Id, log.OwnerId, log); } @IsTest static void it_should_set_owner_when_default_configured_with_queue_developer_name() { + setupConfigurations(); User currentUser = new User(Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); - Group expectedLogOwnerQueue = LoggerTestUtils.insertQueue(Schema.Log__c.SObjectType); + Group expectedLogOwnerQueue = LoggerMockDataCreator.insertQueue('Some_Log_Queue', Schema.Log__c.SObjectType); LoggerSettings__c currentUserSettings = Logger.getUserSettings(currentUser); currentUserSettings.DefaultLogOwner__c = expectedLogOwnerQueue.DeveloperName; insert currentUserSettings; - Log__c log = new Log__c(LoggedBy__c = currentUser.Id, OwnerId = currentUser.Id, TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, OwnerId FROM Log__c WHERE Id = :log.Id]; System.assertEquals(expectedLogOwnerQueue.Id, log.OwnerId, log); } @IsTest static void it_should_use_current_user_as_owner_when_no_default_configured() { + setupConfigurations(); User currentUser = new User(Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); LoggerSettings__c currentUserSettings = Logger.getUserSettings(currentUser); currentUserSettings.DefaultLogOwner__c = null; insert currentUserSettings; - Log__c log = new Log__c(LoggedBy__c = currentUser.Id, OwnerId = currentUser.Id, TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, OwnerId FROM Log__c WHERE Id = :log.Id]; System.assertEquals(UserInfo.getUserId(), log.OwnerId, log); } @IsTest - static void it_should_keep_existing_retention_date_when_populated() { + static void it_should_keep_existing_retention_details_when_populated() { + setupConfigurations(); Integer defaultDaysToRetainLog = 10; - + String defaultLogPurgeAction = 'Some Action'; LoggerSettings__c settings = Logger.getUserSettings(); settings.DefaultNumberOfDaysToRetainLogs__c = defaultDaysToRetainLog; + settings.DefaultLogPurgeAction__c = defaultLogPurgeAction; upsert settings; - - Test.startTest(); - Integer specifiedDaysToRetainLog = 50; Date retentionDate = System.today().addDays(specifiedDaysToRetainLog); - Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), LogRetentionDate__c = retentionDate, TransactionId__c = '1234'); - insert log; + String specifiedLogPurgeAction = 'A Different Action'; + Log__c log = new Log__c( + LoggedBy__c = UserInfo.getUserId(), + LogPurgeAction__c = specifiedLogPurgeAction, + LogRetentionDate__c = retentionDate, + TransactionId__c = '1234' + ); - Test.stopTest(); + LoggerDataStore.getDatabase().insertRecord(log); - log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + log = [SELECT Id, LogPurgeAction__c, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; System.assertNotEquals(defaultDaysToRetainLog, specifiedDaysToRetainLog); + System.assertNotEquals(defaultLogPurgeAction, specifiedLogPurgeAction); + System.assertEquals(specifiedLogPurgeAction, log.LogPurgeAction__c); System.assertEquals(retentionDate, log.LogRetentionDate__c); } @IsTest - static void it_should_set_retention_date_when_configured_via_logger_settings() { + static void it_should_set_retention_details_when_configured_via_logger_settings() { + setupConfigurations(); Integer daysToRetainLog = 90; Date expectedRetentionDate = System.today().addDays(daysToRetainLog); + String expectedLogPurgeAction = 'Some Action'; Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c = daysToRetainLog; + Logger.getUserSettings().DefaultLogPurgeAction__c = expectedLogPurgeAction; upsert Logger.getUserSettings(); + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(LoggedBy__c = UserInfo.getUserId())).populateRequiredFields().getRecord(); - Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); - log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + log = [SELECT Id, LogPurgeAction__c, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(expectedLogPurgeAction, log.LogPurgeAction__c); System.assertEquals(expectedRetentionDate, log.LogRetentionDate__c); } @IsTest - static void it_should_set_retention_date_when_configured_via_scenario_rules() { + static void it_should_set_retention_details_when_configured_via_scenario_rules() { + setupConfigurations(); Integer defaultDaysToRetainLog = 1; Integer daysToRetainLog = 90; Date expectedRetentionDate = System.today().addDays(daysToRetainLog); Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c = defaultDaysToRetainLog; upsert Logger.getUserSettings(); String transactionScenarioName = 'some scenario'; - LogScenarioRule__mdt scenarioRule = new LogScenarioRule__mdt(Scenario__c = transactionScenarioName, NumberOfDaysToRetainLogs__c = daysToRetainLog); - LogHandler.setMockScenarioRule(scenarioRule); + LoggerTestConfigurator.setMock(new LogScenarioRule__mdt(Scenario__c = transactionScenarioName, NumberOfDaysToRetainLogs__c = daysToRetainLog)); + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + log.Scenario__c = transactionScenarioName; - Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), Scenario__c = transactionScenarioName, TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); log = [SELECT Id, Scenario__c, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(expectedRetentionDate, log.LogRetentionDate__c); } @IsTest - static void it_should_have_null_retention_date_when_no_retention_configured() { + static void it_should_have_null_retention_details_when_no_retention_configured() { + setupConfigurations(); LoggerSettings__c settings = Logger.getUserSettings(); + insert settings; settings.DefaultNumberOfDaysToRetainLogs__c = null; + settings.DefaultLogPurgeAction__c = null; upsert settings; - - Test.startTest(); - - Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), TransactionId__c = '1234'); - insert log; - - Test.stopTest(); - - log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + settings = [SELECT Id, DefaultLogPurgeAction__c FROM LoggerSettings__c WHERE Id = :settings.Id]; + System.assertEquals(null, settings.DefaultLogPurgeAction__c); + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + log.LoggedBy__c = UserInfo.getUserId(); + log.ProfileId__c = UserInfo.getProfileId(); + + LoggerDataStore.getDatabase().insertRecord(log); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + String defaultLogPurgeActionPicklistValue = getDefaultPicklistValue(Schema.Log__c.LogPurgeAction__c); + log = [SELECT Id, LogPurgeAction__c, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(defaultLogPurgeActionPicklistValue, log.LogPurgeAction__c); System.assertEquals(null, log.LogRetentionDate__c); } @IsTest - static void it_should_set_priority_to_high_when_there_are_errors() { - Log__c log = new Log__c(Priority__c = LOW_PRIORITY, TransactionId__c = '1234'); - insert log; + static void it_should_set_priority_to_high_when_there_is_an_error_log_entry() { + setupConfigurations(); + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + log.Priority__c = LOW_PRIORITY; + LoggerDataStore.getDatabase().insertRecord(log); log = [SELECT Id, Priority__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(LOW_PRIORITY, log.Priority__c); - Test.startTest(); - insert new LogEntry__c(Log__c = log.Id, LoggingLevel__c = 'ERROR'); - Test.stopTest(); + insert LoggerMockDataCreator.createDataBuilder(new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.ERROR.name())) + .populateRequiredFields() + .getRecord(); + System.assertEquals( + 4, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed four times - two times for BEFORE_INSERT/AFTER_INSERT' + + ' and two more times for BEFORE_UPDATE/AFTER_UPDATE (triggered indirectly by the insert of a related LogEntry__c record)' + ); log = [SELECT Id, Priority__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(HIGH_PRIORITY, log.Priority__c); } @IsTest - static void it_should_set_priority_to_medium_when_there_are_warnings() { + static void it_should_set_priority_to_medium_when_there_is_a_warn_log_entry() { + setupConfigurations(); Log__c log = new Log__c(Priority__c = LOW_PRIORITY, TransactionId__c = '1234'); - insert log; + LoggerDataStore.getDatabase().insertRecord(log); log = [SELECT Id, Priority__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(LOW_PRIORITY, log.Priority__c); - Test.startTest(); insert new LogEntry__c(Log__c = log.Id, LoggingLevel__c = 'WARN'); - Test.stopTest(); + System.assertEquals( + 4, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed four times - two times for BEFORE_INSERT/AFTER_INSERT' + + ' and two more times for BEFORE_UPDATE/AFTER_UPDATE (triggered indirectly by the insert of a related LogEntry__c record)' + ); log = [SELECT Id, Priority__c FROM Log__c WHERE Id = :log.Id]; System.assertEquals(MEDIUM_PRIORITY, log.Priority__c); } @IsTest static void it_should_grant_read_access_to_user_when_access_level_is_read() { + setupConfigurations(); LoggerSettings__c settings = Logger.getUserSettings(); settings.DefaultLogShareAccessLevel__c = 'Read'; upsert settings; - - Test.startTest(); Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), TransactionId__c = '1234'); - insert log; - Test.stopTest(); - List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; + LoggerDataStore.getDatabase().insertRecord(log); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; System.assertEquals(1, logShares.size(), logShares); System.assertEquals('Read', logShares.get(0).AccessLevel); System.assertEquals(log.Id, logShares.get(0).ParentId); @@ -292,17 +368,20 @@ private class LogHandler_Tests { @IsTest static void it_should_grant_edit_access_to_user_when_access_level_is_edit() { + setupConfigurations(); LoggerSettings__c settings = Logger.getUserSettings(); settings.DefaultLogShareAccessLevel__c = 'Edit'; upsert settings; - - Test.startTest(); Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), TransactionId__c = '1234'); - insert log; - Test.stopTest(); - List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; + LoggerDataStore.getDatabase().insertRecord(log); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; System.assertEquals(1, logShares.size(), logShares); System.assertEquals('Edit', logShares.get(0).AccessLevel); System.assertEquals(log.Id, logShares.get(0).ParentId); @@ -312,52 +391,57 @@ private class LogHandler_Tests { @IsTest static void it_should_not_grant_access_to_user_when_access_level_is_null() { + setupConfigurations(); LoggerSettings__c settings = Logger.getUserSettings(); settings.DefaultLogShareAccessLevel__c = null; upsert settings; - - Test.startTest(); Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), TransactionId__c = '1234'); - insert log; - Test.stopTest(); - List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; + LoggerDataStore.getDatabase().insertRecord(log); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; System.assertEquals(0, logShares.size(), logShares); } @IsTest static void it_should_not_grant_access_to_user_when_access_level_is_an_unknown_string() { + setupConfigurations(); LoggerSettings__c settings = Logger.getUserSettings(); settings.DefaultLogShareAccessLevel__c = 'FAKE LEVEL'; upsert settings; - - Test.startTest(); Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), TransactionId__c = '1234'); - insert log; - Test.stopTest(); - List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; + LoggerDataStore.getDatabase().insertRecord(log); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.Log__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + List logShares = [SELECT AccessLevel, ParentId, RowCause, UserOrGroupId FROM Log__Share WHERE ParentId = :log.Id AND AccessLevel != 'All']; System.assertEquals(0, logShares.size(), logShares); } - @IsTest - static void it_should_run_apex_plugin_when_configured() { - String expectedProfileName = 'Some String'; - - Test.startTest(); - - // Use the mock configurations - LoggerPlugin__mdt plugin = new LoggerPlugin__mdt(PluginType__c = 'Apex', PluginApiName__c = LogPluginTest.class.getName()); - LoggerSObjectHandler.setMockPlugin(Schema.Log__c.SObjectType, plugin); - - Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), TransactionId__c = '1234'); - insert log; + private static String getDefaultPicklistValue(Schema.SObjectField field) { + List picklistEntries = field.getDescribe().getPicklistValues(); + for (Schema.PicklistEntry picklistEntry : picklistEntries) { + if (picklistEntry.isDefaultValue()) { + return picklistEntry.getValue(); + } + } + return null; + } - log = [SELECT Id, ProfileName__c FROM Log__c WHERE Id = :log.Id]; - System.assertEquals(expectedProfileName, log.ProfileName__c); + private static void setupConfigurations() { + LogStatus__mdt openStatus = new LogStatus__mdt(IsActive__c = true, MasterLabel = FIRST_STATUS, IsClosed__c = false, IsResolved__c = false); + LoggerTestConfigurator.setMock(openStatus); - Test.stopTest(); + LogStatus__mdt closedStatus = new LogStatus__mdt(IsActive__c = true, MasterLabel = SECOND_STATUS, IsClosed__c = true, IsResolved__c = true); + LoggerTestConfigurator.setMock(closedStatus); } } diff --git a/nebula-logger/core/tests/log-management/classes/LogMassDeleteExtension_Tests.cls b/nebula-logger/core/tests/log-management/classes/LogMassDeleteExtension_Tests.cls index 3a6856854..b0e173215 100644 --- a/nebula-logger/core/tests/log-management/classes/LogMassDeleteExtension_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LogMassDeleteExtension_Tests.cls @@ -4,13 +4,16 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=false) private class LogMassDeleteExtension_Tests { @TestSetup static void setupData() { + LoggerSObjectHandler.shouldExecute(false); + List logs = new List(); for (Integer i = 0; i < 10; i++) { Log__c log = new Log__c(TransactionId__c = 'TXN-' + i); + LoggerMockDataCreator.createDataBuilder(log).populateRequiredFields().getRecord(); logs.add(log); } insert logs; @@ -19,12 +22,10 @@ private class LogMassDeleteExtension_Tests { @IsTest static void it_should_return_deletable_logs() { List logs = [SELECT Id, Name FROM Log__c]; - List logIds = new List(); for (Log__c selectedLog : logs) { logIds.add(selectedLog.Id); } - List expectedDeletableLogs = new List(); for (UserRecordAccess recordAccess : [ SELECT RecordId @@ -33,20 +34,14 @@ private class LogMassDeleteExtension_Tests { ]) { expectedDeletableLogs.add(new Log__c(Id = recordAccess.RecordId)); } - ApexPages.StandardSetController controller = new ApexPages.StandardSetController(logs); controller.setSelected(logs); - PageReference pageReference = Page.LogMassDelete; Test.setCurrentPage(pageReference); - Test.startTest(); - LogMassDeleteExtension extension = new LogMassDeleteExtension(controller); List returnedDeletableLogs = extension.getDeletableLogs(); - Test.stopTest(); - System.assertEquals(expectedDeletableLogs.size(), returnedDeletableLogs.size()); } @@ -63,21 +58,14 @@ private class LogMassDeleteExtension_Tests { logsToKeep.add(logs.get(i)); } } - ApexPages.StandardSetController controller = new ApexPages.StandardSetController(logs); controller.setSelected(logsToDelete); - PageReference pageReference = Page.LogMassDelete; Test.setCurrentPage(pageReference); - Test.startTest(); - LogMassDeleteExtension extension = new LogMassDeleteExtension(controller); extension.deleteSelectedLogs(); - Test.stopTest(); - - // Verify that only the selected logs were deleted logsToDelete = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToDelete ALL ROWS]; for (Log__c log : logsToDelete) { System.assertEquals(true, log.IsDeleted, log); diff --git a/nebula-logger/core/tests/log-management/classes/LoggerBatchableContext_Tests.cls b/nebula-logger/core/tests/log-management/classes/LoggerBatchableContext_Tests.cls new file mode 100644 index 000000000..8465891e6 --- /dev/null +++ b/nebula-logger/core/tests/log-management/classes/LoggerBatchableContext_Tests.cls @@ -0,0 +1,32 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LoggerBatchableContext_Tests { + @IsTest + static void it_constructs_instance_when_both_parameters_provided() { + Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + Schema.SObjectType sobjectType = Schema.LogEntryEvent__e.SObjectType; + + LoggerBatchableContext context = new LoggerBatchableContext(mockBatchableContext, sobjectType); + + System.assertEquals(mockBatchableContext, context.batchableContext); + System.assertEquals(sobjectType, context.sobjectType); + System.assertEquals(sobjectType.getDescribe().getName(), context.sobjectTypeName); + } + + @IsTest + static void it_constructs_instance_when_neither_parameters_provided() { + Database.BatchableContext nullBatchableContext = null; + Schema.SObjectType nullSObjectType = null; + + LoggerBatchableContext context = new LoggerBatchableContext(nullBatchableContext, nullSObjectType); + + System.assertEquals(null, context.batchableContext); + System.assertEquals(null, context.sobjectType); + System.assertEquals(null, context.sobjectTypeName); + } +} diff --git a/nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls-meta.xml b/nebula-logger/core/tests/log-management/classes/LoggerBatchableContext_Tests.cls-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls-meta.xml rename to nebula-logger/core/tests/log-management/classes/LoggerBatchableContext_Tests.cls-meta.xml diff --git a/nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls b/nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls new file mode 100644 index 000000000..8df5ae14b --- /dev/null +++ b/nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls @@ -0,0 +1,134 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LoggerEmailSender_Tests { + @IsTest + static void it_should_filter_apex_error_recipients() { + ApexEmailNotification emailListNotification = new ApexEmailNotification(Email = 'hello@test.com;fake.person@not.real.com.biz', UserId = null); + ApexEmailNotification userNotification = new ApexEmailNotification(Email = null, UserId = UserInfo.getUserId()); + ApexEmailNotification invalidNotification = new ApexEmailNotification(Email = null, UserId = null); + LoggerEmailSender.MOCK_NOTIFICATIONS.addAll(new List{ emailListNotification, userNotification, invalidNotification }); + + List returnedRecipients = LoggerEmailSender.CACHED_APEX_ERROR_RECIPIENTS; + + System.assertEquals(3, returnedRecipients.size(), 'Should have returned 3 recipients: 1 for the user ID, and 2 for the email addresses'); + for (String recipient : returnedRecipients) { + Boolean matchesUserNotification = String.valueOf(userNotification.UserId) == recipient; + Boolean matchesEmailListNotification = new Set(emailListNotification.Email.split(';')).contains(recipient.trim()); + System.assertEquals( + true, + matchesUserNotification || matchesEmailListNotification, + 'Returned recipient ' + + recipient + + ' should match either the user notification or the email list notification\n' + + JSON.serializePretty(LoggerEmailSender.MOCK_NOTIFICATIONS) + ); + } + } + + @IsTest + static void it_should_send_email_notification_for_saveResult_errors_when_enabled() { + LoggerEmailSender.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); + + // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (intentionally) fail + List saveResultsWithErrors = new List{ LoggerMockDataCreator.createDatabaseSaveResult(false) }; + LoggerEmailSender.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResultsWithErrors); + + System.assertEquals( + true, + LoggerEmailSender.SENT_EMAILS.get(0).getHtmlBody().contains(saveResultsWithErrors.get(0).errors.get(0).getMessage()), + 'Email message should contain SaveResult error message' + ); + if (LoggerEmailSender.IS_EMAIL_DELIVERABILITY_ENABLED == true) { + System.assertEquals(1, Limits.getEmailInvocations(), 'Email should have been sent'); + } else { + System.assertEquals(0, Limits.getEmailInvocations(), 'Deliverability is not currently enabled'); + } + } + + @IsTest + static void it_should_not_send_email_notification_for_saveResult_errors_when_no_recipients_configured() { + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'true')); + System.assertEquals(true, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); + LoggerEmailSender.CACHED_APEX_ERROR_RECIPIENTS.clear(); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); + + // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (intentionally) fail + List saveResultsWithErrors = new List{ LoggerMockDataCreator.createDatabaseSaveResult(false) }; + LoggerEmailSender.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResultsWithErrors); + + System.assertEquals(true, LoggerEmailSender.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); + } + + @IsTest + static void it_should_not_send_email_notification_for_saveResult_errors_when_disabled() { + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'false')); + System.assertEquals(false, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); + LoggerEmailSender.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); + + // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (intentionally) fail + List saveResultsWithErrors = new List{ LoggerMockDataCreator.createDatabaseSaveResult(false) }; + LoggerEmailSender.sendErrorEmail(Schema.LogEntry__c.SObjectType, saveResultsWithErrors); + + System.assertEquals(true, LoggerEmailSender.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); + } + + @IsTest + static void it_should_send_email_notification_for_upsertResult_errors_when_enabled() { + LoggerEmailSender.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); + + // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (intentionally) fail + List upsertResultsWithErrors = Database.upsert(new List{ new LogEntry__c() }, false); + LoggerEmailSender.sendErrorEmail(Schema.LogEntry__c.SObjectType, upsertResultsWithErrors); + + System.assertEquals( + true, + LoggerEmailSender.SENT_EMAILS.get(0).getHtmlBody().contains(upsertResultsWithErrors.get(0).errors.get(0).getMessage()), + 'Email message should contain UpsertResult error message' + ); + if (LoggerEmailSender.IS_EMAIL_DELIVERABILITY_ENABLED == true) { + System.assertEquals(1, Limits.getEmailInvocations(), 'Email should have been sent'); + } else { + System.assertEquals(0, Limits.getEmailInvocations(), 'Deliverability is not currently enabled'); + } + } + + @IsTest + static void it_should_not_send_email_notification_for_upsertResult_errors_when_no_recipients_configured() { + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'true')); + System.assertEquals(true, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); + LoggerEmailSender.CACHED_APEX_ERROR_RECIPIENTS.clear(); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); + + // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (intentionally) fail + List upsertResultsWithErrors = Database.upsert(new List{ new LogEntry__c() }, false); + LoggerEmailSender.sendErrorEmail(Schema.LogEntry__c.SObjectType, upsertResultsWithErrors); + + System.assertEquals(true, LoggerEmailSender.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); + } + + @IsTest + static void it_should_not_send_email_notification_for_upsertResult_errors_when_disabled() { + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'SendErrorEmailNotifications', Value__c = 'false')); + System.assertEquals(false, LoggerParameter.SEND_ERROR_EMAIL_NOTIFICATIONS); + LoggerEmailSender.CACHED_APEX_ERROR_RECIPIENTS.add(UserInfo.getUserId()); + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent yet'); + + // LogEntry__c requires a Log__c parent record, so inserting a LogEntry__c with no fields set will (intentionally) fail + List upsertResultsWithErrors = Database.upsert(new List{ new LogEntry__c() }, false); + LoggerEmailSender.sendErrorEmail(Schema.LogEntry__c.SObjectType, upsertResultsWithErrors); + + System.assertEquals(0, Limits.getEmailInvocations(), 'No emails should have been sent'); + System.assertEquals(true, LoggerEmailSender.SENT_EMAILS.isEmpty(), 'No email messages should have been generated'); + } +} diff --git a/nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls-meta.xml b/nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/core/tests/log-management/classes/LoggerEmailSender_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/core/tests/log-management/classes/LoggerSObjectMetadata_Tests.cls b/nebula-logger/core/tests/log-management/classes/LoggerSObjectMetadata_Tests.cls index 50fc37c60..803773c8e 100644 --- a/nebula-logger/core/tests/log-management/classes/LoggerSObjectMetadata_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LoggerSObjectMetadata_Tests.cls @@ -3,19 +3,28 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// +/** + * @description Test class for LoggerSObjectMetadata, used for dynamically converting SObject describe details + * into `@AuraEnabled` objects that can be used in LWCs + * To demonstrate that the class is standalone/would work with any SObject, these test methods + * all use the `User` SObject + */ @SuppressWarnings('PMD.ApexDoc, PMD.ApexUnitTestClassShouldHaveAsserts, PMD.MethodNamingConventions') -@IsTest +@IsTest(IsParallel=true) private class LoggerSObjectMetadata_Tests { @IsTest - static void it_should_return_schema_for_logEntryEvent() { - LoggerSObjectMetadata.SObjectSchema sobjectSchema = LoggerSObjectMetadata.getLogEntryEventSchema(); - validateSObjectDetails(Schema.LogEntryEvent__e.SObjectType, sobjectSchema); + static void it_should_return_schema_for_specified_sobject_api_name() { + Schema.SObjectType userSObjectType = Schema.User.SObjectType; + String userSObjectApiName = userSObjectType.getDescribe().getName(); + LoggerSObjectMetadata.SObjectSchema sobjectSchema = LoggerSObjectMetadata.getSchemaForName(userSObjectApiName); + validateSObjectDetails(userSObjectType, sobjectSchema); } @IsTest - static void it_should_return_schema_for_loggerSettings() { - LoggerSObjectMetadata.SObjectSchema sobjectSchema = LoggerSObjectMetadata.getLoggerSettingsSchema(); - validateSObjectDetails(Schema.LoggerSettings__c.SObjectType, sobjectSchema); + static void it_should_return_schema_for_specified_sobject_type() { + Schema.SObjectType userSObjectType = Schema.User.SObjectType; + LoggerSObjectMetadata.SObjectSchema sobjectSchema = LoggerSObjectMetadata.getSchema(userSObjectType); + validateSObjectDetails(userSObjectType, sobjectSchema); } private static void validateSObjectDetails(SObjectType sobjectType, LoggerSObjectMetadata.SObjectSchema sobjectSchema) { diff --git a/nebula-logger/core/tests/log-management/classes/LoggerSettingsController_Tests.cls b/nebula-logger/core/tests/log-management/classes/LoggerSettingsController_Tests.cls index 474eb964e..38869a1ae 100644 --- a/nebula-logger/core/tests/log-management/classes/LoggerSettingsController_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LoggerSettingsController_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions') -@IsTest +@IsTest(IsParallel=false) private class LoggerSettingsController_Tests { @IsTest static void it_should_return_loggingLevel_picklist_options() { @@ -22,6 +22,34 @@ private class LoggerSettingsController_Tests { } } + @IsTest + static void it_should_return_logPurgeAction_picklist_options() { + List fakeLogPurgeActions = new List{ 'A_FAKE_PURGE_ACTION', 'ANOTHER_ONE', 'SOME_OTHER_PURGE_ACTION' }; + for (String fakeLogPurgeAction : fakeLogPurgeActions) { + LoggerTestConfigurator.setMock( + new LoggerParameter__mdt( + DeveloperName = LoggerSettingsController.CUSTOM_LOG_PURGE_ACTION_PREFIX + fakeLogPurgeAction, + Value__c = fakeLogPurgeAction + ) + ); + } + + List picklistOptions = LoggerSettingsController.getPicklistOptions().purgeActionOptions; + + Set expectedLogPurgeActions = new Set{ LoggerSettingsController.DELETE_LOG_PURGE_ACTION }; + expectedLogPurgeActions.addAll(fakeLogPurgeActions); + Integer expectedLogPurgeActionsSize = expectedLogPurgeActions.size() + 1; // '--NONE--' is automatically included + System.assertEquals(expectedLogPurgeActionsSize, picklistOptions.size()); + for (LoggerSettingsController.PicklistOption picklistOption : picklistOptions) { + if (String.isBlank(picklistOption.value) == true) { + System.assertEquals('--None--', picklistOption.label); + } else { + System.assertEquals(picklistOption.value, picklistOption.label); + System.assertEquals(true, expectedLogPurgeActions.contains(picklistOption.value)); + } + } + } + @IsTest static void it_should_return_loggerSaveMethod_picklist_options() { Integer expectedLoggerSaveMethodSize = Logger.SaveMethod.values().size() + 1; // '--NONE--' is automatically included @@ -33,7 +61,63 @@ private class LoggerSettingsController_Tests { } else { System.assertEquals(picklistOption.value, picklistOption.label); Logger.SaveMethod matchingLoggerSaveMethod = Logger.SaveMethod.valueOf(picklistOption.value); - System.assertEquals(matchingLoggerSaveMethod.name(), picklistOption.label); + System.assertEquals(matchingLoggerSaveMethod.name(), picklistOption.value); + } + } + } + + @IsTest + static void it_should_append_custom_platformEventStorageLocation_picklist_options_when_configured() { + List fakeStorageLocations = new List{ 'NOT_A_REAL_STORAGE_LOCATION', 'OR_MAYBE_IT_IS_A_REAL_ONE', 'BUT_MAYBE_JUST_A_TEST', 'PROBABLY' }; + for (String fakeStorageLocation : fakeStorageLocations) { + LoggerTestConfigurator.setMock( + new LoggerParameter__mdt( + DeveloperName = LoggerSettingsController.CUSTOM_STORAGE_LOCATION_PREFIX + fakeStorageLocation, + Value__c = fakeStorageLocation + ) + ); + } + + List picklistOptions = LoggerSettingsController.getPicklistOptions().platformEventStorageLocationOptions; + + Set expectedSaveMethods = new Set{ LoggerSettingsController.CUSTOM_OBJECTS_STORAGE_LOCATION }; + expectedSaveMethods.addAll(fakeStorageLocations); + Integer expectedPlatformEventStorageLocationSize = expectedSaveMethods.size() + 1; // '--NONE--' is automatically included + System.assertEquals(expectedPlatformEventStorageLocationSize, picklistOptions.size(), picklistOptions); + for (LoggerSettingsController.PicklistOption picklistOption : picklistOptions) { + if (String.isBlank(picklistOption.value) == true) { + System.assertEquals('--None--', picklistOption.label); + } else { + System.assertEquals(picklistOption.value, picklistOption.label, 'picklistOption==' + picklistOption); + System.assertEquals(true, expectedSaveMethods.contains(picklistOption.value), 'picklistOption==' + picklistOption); + } + } + } + + @IsTest + static void it_should_append_custom_loggerSaveMethod_picklist_options_when_configured() { + List fakeSaveMethods = new List{ 'A_FAKE_SAVE_METHOD', 'ANOTHER_ONE', 'SOME_OTHER_SAVE_METHOD' }; + for (String fakeSaveMethod : fakeSaveMethods) { + LoggerTestConfigurator.setMock( + new LoggerParameter__mdt(DeveloperName = LoggerSettingsController.CUSTOM_SAVE_METHOD_PREFIX + fakeSaveMethod, Value__c = fakeSaveMethod) + ); + } + + List picklistOptions = LoggerSettingsController.getPicklistOptions().saveMethodOptions; + + Set expectedSaveMethods = new Set(); + expectedSaveMethods.addAll(fakeSaveMethods); + for (Logger.SaveMethod saveMethod : Logger.SaveMethod.values()) { + expectedSaveMethods.add(saveMethod.name()); + } + Integer expectedLoggerSaveMethodSize = expectedSaveMethods.size() + 1; // '--NONE--' is automatically included + System.assertEquals(expectedLoggerSaveMethodSize, picklistOptions.size(), picklistOptions); + for (LoggerSettingsController.PicklistOption picklistOption : picklistOptions) { + if (String.isBlank(picklistOption.value) == true) { + System.assertEquals('--None--', picklistOption.label); + } else { + System.assertEquals(picklistOption.value, picklistOption.label, 'picklistOption==' + picklistOption); + System.assertEquals(true, expectedSaveMethods.contains(picklistOption.value), 'picklistOption==' + picklistOption); } } } @@ -169,7 +253,7 @@ private class LoggerSettingsController_Tests { LoggerSettings__c invalidRecord = null; String expectedExceptionMessage; try { - delete invalidRecord; + LoggerDataStore.getDatabase().deleteRecord(invalidRecord); } catch (Exception ex) { expectedExceptionMessage = ex.getMessage(); } @@ -266,6 +350,7 @@ private class LoggerSettingsController_Tests { CreatedDate, DefaultLogShareAccessLevel__c, DefaultNumberOfDaysToRetainLogs__c, + DefaultPlatformEventStorageLocation__c, DefaultSaveMethod__c, Id, IsAnonymousModeEnabled__c, @@ -273,7 +358,6 @@ private class LoggerSettingsController_Tests { IsDataMaskingEnabled__c, IsEnabled__c, IsJavaScriptConsoleLoggingEnabled__c, - IsPlatformEventStorageEnabled__c, IsRecordFieldStrippingEnabled__c, IsSavingEnabled__c, LastModifiedBy.Username, diff --git a/nebula-logger/core/tests/log-management/classes/LoggerTagHandler_Tests.cls b/nebula-logger/core/tests/log-management/classes/LoggerTagHandler_Tests.cls index d38f03144..25566ce47 100644 --- a/nebula-logger/core/tests/log-management/classes/LoggerTagHandler_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/LoggerTagHandler_Tests.cls @@ -4,29 +4,83 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LoggerTagHandler_Tests { @IsTest - static void it_should_set_unique_id_when_enabled() { - LoggerTag__c tag = new LoggerTag__c(Name = 'some tag'); - insert tag; + static void it_should_return_the_loggerTag_sobjectType() { + System.assertEquals(Schema.LoggerTag__c.SObjectType, new LoggerTagHandler().getSObjectType()); + } + + @IsTest + static void it_should_not_run_when_disabled_via_configuration() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LoggerTestConfigurator.getSObjectHandlerConfiguration(Schema.LoggerTag__c.SObjectType).IsEnabled__c = false; + LoggerTag__c tag = (LoggerTag__c) LoggerMockDataCreator.createDataBuilder(Schema.LoggerTag__c.SObjectType).populateRequiredFields().getRecord(); + + LoggerDataStore.getDatabase().insertRecord(tag); + + System.assertEquals( + 0, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LoggerTag__c.SObjectType).size(), + 'Handler class should not have executed' + ); + } + + @IsTest + static void it_should_set_unique_id_on_insert() { + LoggerTag__c tag = (LoggerTag__c) LoggerMockDataCreator.createDataBuilder(Schema.LoggerTag__c.SObjectType).populateRequiredFields().getRecord(); + tag.Name = 'Some tag name'; + LoggerDataStore.getDatabase().insertRecord(tag); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LoggerTag__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + tag = [SELECT Id, Name, UniqueId__c FROM LoggerTag__c WHERE Id = :tag.Id]; + System.assertEquals(tag.Name, tag.UniqueId__c); + } + + @IsTest + static void it_should_set_unique_id_on_update() { + LoggerTag__c tag = (LoggerTag__c) LoggerMockDataCreator.createDataBuilder(Schema.LoggerTag__c.SObjectType).populateRequiredFields().getRecord(); + tag.Name = 'Some tag name'; + LoggerDataStore.getDatabase().insertRecord(tag); + + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LoggerTag__c.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); tag = [SELECT Id, Name, UniqueId__c FROM LoggerTag__c WHERE Id = :tag.Id]; System.assertEquals(tag.Name, tag.UniqueId__c); } @IsTest static void it_should_not_insert_duplicate_tag() { - LoggerTag__c tag = new LoggerTag__c(Name = 'some tag'); - insert tag; + LoggerTag__c tag = (LoggerTag__c) LoggerMockDataCreator.createDataBuilder(Schema.LoggerTag__c.SObjectType).populateRequiredFields().getRecord(); + tag.Name = 'Some tag name'; + LoggerDataStore.getDatabase().insertRecord(tag); + LoggerTag__c duplicateTag = (LoggerTag__c) LoggerMockDataCreator.createDataBuilder(new LoggerTag__c(Name = tag.Name)) + .populateRequiredFields() + .getRecord(); + Exception thrownException; - LoggerTag__c duplicateTag = new LoggerTag__c(Name = tag.Name); try { insert duplicateTag; System.assert(false, 'Exception expected on previous line'); } catch (Exception ex) { - String expectedDuplicateError = 'DUPLICATE_VALUE'; - System.assert(ex.getMessage().contains(expectedDuplicateError), ex.getMessage()); + thrownException = ex; } + + System.assertEquals( + 3, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LoggerTag__c.SObjectType).size(), + 'Handler class should have executed three times - once for BEFORE_INSERT and once for AFTER_INSERT for the first record,' + + ' and once for BEFORE_INSERT on the errored duplicate' + ); + System.assertNotEquals(null, thrownException, 'An exception should have been thrown'); + String expectedDuplicateError = 'DUPLICATE_VALUE'; + System.assert(thrownException.getMessage().contains(expectedDuplicateError), thrownException.getMessage()); } } diff --git a/nebula-logger/core/tests/log-management/classes/RelatedLogEntriesController_Tests.cls b/nebula-logger/core/tests/log-management/classes/RelatedLogEntriesController_Tests.cls index ae5eafcc4..9efd31e79 100644 --- a/nebula-logger/core/tests/log-management/classes/RelatedLogEntriesController_Tests.cls +++ b/nebula-logger/core/tests/log-management/classes/RelatedLogEntriesController_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class RelatedLogEntriesController_Tests { private static final Integer TOTAL_LOG_ENTRIES = 10; private static final Integer TOTAL_RELATED_LOG_ENTRIES = 7; @@ -19,16 +19,18 @@ private class RelatedLogEntriesController_Tests { @TestSetup static void setupData() { - Log__c log = new Log__c(TransactionId__c = '1234'); + LoggerSObjectHandler.shouldExecute(false); + + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); insert log; List logEntries = new List(); for (Integer i = 0; i < TOTAL_LOG_ENTRIES; i++) { LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, Message__c = 'test ' + i); - if (i <= TOTAL_RELATED_LOG_ENTRIES) { logEntry.RecordId__c = UserInfo.getUserId(); } + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); logEntries.add(logEntry); } diff --git a/nebula-logger/core/tests/log-management/testSuites/LoggerLogManagement.testSuite-meta.xml b/nebula-logger/core/tests/log-management/testSuites/LoggerLogManagement.testSuite-meta.xml index 515914a5a..1007c5ecf 100644 --- a/nebula-logger/core/tests/log-management/testSuites/LoggerLogManagement.testSuite-meta.xml +++ b/nebula-logger/core/tests/log-management/testSuites/LoggerLogManagement.testSuite-meta.xml @@ -6,6 +6,8 @@ LogEntryFieldSetPicklist_Tests LogEntryHandler_Tests LogEntryTagHandler_Tests + LoggerBatchableContext_Tests + LoggerEmailSender_Tests LoggerSettingsController_Tests LoggerSObjectMetadata_Tests LoggerTagHandler_Tests diff --git a/nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls b/nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls new file mode 100644 index 000000000..6a9dc9cd3 --- /dev/null +++ b/nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls @@ -0,0 +1,162 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Test Utilities + * @description Utility class used to help with setting up Nebula Logger's configurations within a test context. + * These methods are specific to metadata implemented within Nebula Logger. + * These methods can be used when writing Apex tests for plugins. + * @see LoggerMockDataCreator + * @see LoggerMockDataStore + */ +@IsTest +public without sharing class LoggerTestConfigurator { + private static final Map PERMISSION_SETS_BY_NAME = queryPermissionSets(); + + /** + * @description Assigns the permission set `LoggerAdmin` to the specified user ID + * @param userId The ID of the user that should be assigned the permission set + */ + public static void assignAdminPermissionSet(Id userId) { + assignPermissionSet(userId, PERMISSION_SETS_BY_NAME.get('LoggerAdmin')); + } + + /** + * @description Assigns the permission set `LoggerLogViewer` to the specified user ID + * @param userId The ID of the user that should be assigned the permission set + */ + public static void assignLogViewerPermissionSet(Id userId) { + assignPermissionSet(userId, PERMISSION_SETS_BY_NAME.get('LoggerLogViewer')); + } + + /** + * @description Assigns the permission set `LoggerEndUser` to the specified user ID + * @param userId The ID of the user that should be assigned the permission set + */ + public static void assignEndUserPermissionSet(Id userId) { + assignPermissionSet(userId, PERMISSION_SETS_BY_NAME.get('LoggerEndUser')); + } + + /** + * @description Assigns the permission set `LoggerLogCreator` to the specified user ID + * @param userId The ID of the user that should be assigned the permission set + */ + public static void assignLogCreatorPermissionSet(Id userId) { + assignPermissionSet(userId, PERMISSION_SETS_BY_NAME.get('LoggerLogCreator')); + } + + /** + * @description Returns an instance of `LoggerSObjectHandler__mdt` that has been built & configured for the specified `SObjectType` + * @param sobjectType The instance `SObjectType` to check for a configured instance of `LoggerSObjectHandler` + * @return The ``LoggerSObjectHandler__mdt` record that has been configured for the specified `SObjectType` (if any) + */ + public static LoggerSObjectHandler__mdt getSObjectHandlerConfiguration(Schema.SObjectType sobjectType) { + return LoggerSObjectHandler.getHandlerConfiguration(sobjectType); + } + + /** + * @description Loads the mock `LogEntryDataMaskRule__mdt` during test execution + * @param mock The mock instance of `LogEntryDataMaskRule__mdt` to load + */ + public static void setMock(LogEntryDataMaskRule__mdt mock) { + LogEntryEventBuilder.setMockDataMaskRule(mock); + } + + /** + * @description Loads the mock `LogEntryTagRule__mdt` during test execution + * @param mock The mock instance of `LogEntryTagRule__mdt` to load + */ + public static void setMock(LogEntryTagRule__mdt mock) { + LogEntryEventHandler.TAG_ASSIGNMENT_RULES.add(mock); + } + + /** + * @description Loads the mock `LoggerParameter__mdt` during test execution + * @param mock The mock instance of `LoggerParameter__mdt` to load + */ + public static void setMock(LoggerParameter__mdt mock) { + LoggerParameter.setMock(mock); + } + + /** + * @description Loads the mock `LoggerPlugin__mdt` during test execution + * @param mock The mock instance of `LoggerPlugin__mdt` to load + */ + public static void setMock(LoggerPlugin__mdt mock) { + LoggerPlugin.setMock(mock); + } + + /** + * @description Loads the mock `LoggerSObjectHandler__mdt` during test execution + * @param mock The mock instance of `LoggerSObjectHandler__mdt` to load + */ + public static void setMock(LoggerSObjectHandler__mdt mock) { + LoggerSObjectHandler.setMock(mock); + } + + /** + * @description Loads the mock `LogScenarioRule__mdt` during test execution + * @param mock The mock instance of `LogScenarioRule__mdt` to load + */ + public static void setMock(LogScenarioRule__mdt mock) { + Logger.setMockScenarioRule(mock); + LogHandler.setMockScenarioRule(mock); + } + + /** + * @description Loads the mock `LogStatus__mdt` during test execution + * @param mock The mock instance of `LogStatus__mdt` to load + */ + public static void setMock(LogStatus__mdt mock) { + LogHandler.setMockLogStatus(mock); + } + + /** + * @description Creates mock instances of `LoggerSObjectHandler__mdt` for each `SObjectType` used by Nebula Logger, + * with each `LoggerSObjectHandler__mdt` automatically set to `IsEnabled__c == true` + */ + public static void setupMockSObjectHandlerConfigurations() { + setupMockSObjectHandlerConfigurations(true); + } + + /** + * @description Creates mock instances of `LoggerSObjectHandler__mdt` for each `SObjectType` used by Nebula Logger, + * with each `LoggerSObjectHandler__mdt` enabled/disabled based on the provided boolean + * @param isEnabled The Boolean value to control if all mock `LoggerSObjectHandler__mdt` records should be enabled (`true`) or disabled (`false`) + */ + public static void setupMockSObjectHandlerConfigurations(Boolean isEnabled) { + Map sobjectTypeToDefaultHandlerApexClass = new Map{ + Schema.Log__c.SObjectType => LogHandler.class, + Schema.LogEntry__c.SObjectType => LogEntryHandler.class, + Schema.LogEntryEvent__e.SObjectType => LogEntryEventHandler.class, + Schema.LogEntryTag__c.SObjectType => LogEntryTagHandler.class, + Schema.LoggerTag__c.SObjectType => LoggerTagHandler.class + }; + for (Schema.SObjectType sobjectType : sobjectTypeToDefaultHandlerApexClass.keySet()) { + setMock( + new LoggerSObjectHandler__mdt( + IsEnabled__c = isEnabled, + SObjectHandlerApexClass__c = sobjectTypeToDefaultHandlerApexClass.get(sobjectType).getName(), + SObjectType__c = sobjectType.getDescribe().getName() + ) + ); + } + } + + // Helper methods + private static void assignPermissionSet(Id userId, PermissionSet permissionSet) { + PermissionSetAssignment permissionSetAssignment = new PermissionSetAssignment(AssigneeId = userId, PermissionSetId = permissionSet.Id); + insert permissionSetAssignment; + } + + private static Map queryPermissionSets() { + List permissionSetNames = new List{ 'LoggerAdmin', 'LoggerLogViewer', 'LoggerEndUser', 'LoggerLogCreator' }; + Map results = new Map(); + for (PermissionSet permissionSet : [SELECT Id, Name FROM PermissionSet WHERE Name IN :permissionSetNames]) { + results.put(permissionSet.Name, permissionSet); + } + return results; + } +} diff --git a/nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls-meta.xml b/nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/core/tests/log-management/utilities/LoggerTestConfigurator.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/core/tests/logger-engine/classes/ComponentLogger_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/ComponentLogger_Tests.cls index 223d7aba5..93a7edd17 100644 --- a/nebula-logger/core/tests/logger-engine/classes/ComponentLogger_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/ComponentLogger_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class ComponentLogger_Tests { @IsTest static void it_should_return_logger_settings() { @@ -13,28 +13,22 @@ private class ComponentLogger_Tests { loggerSettings.LoggingLevel__c = 'FINEST'; upsert loggerSettings; - Test.startTest(); ComponentLogger.ComponentLoggerSettings componentLoggerSettings = ComponentLogger.getSettings(); - Test.stopTest(); System.assertEquals(loggerSettings.IsEnabled__c, componentLoggerSettings.isEnabled); System.assertEquals(loggerSettings.LoggingLevel__c, componentLoggerSettings.userLoggingLevel.name); - LoggingLevel userLoggingLevel = Logger.getLoggingLevel(loggerSettings.LoggingLevel__c); System.assertEquals(userLoggingLevel.name(), componentLoggerSettings.userLoggingLevel.name); System.assertEquals(userLoggingLevel.ordinal(), componentLoggerSettings.userLoggingLevel.ordinal); - for (LoggingLevel currentLoggingLevel : LoggingLevel.values()) { // We don't care about logging level NONE, or the secret/undocumented INTERNAL logging level if (currentLoggingLevel == LoggingLevel.NONE || currentLoggingLevel == LoggingLevel.INTERNAL) { continue; } - System.assert( componentLoggerSettings.supportedLoggingLevels.containsKey(currentLoggingLevel.name()), 'Cmp settings did not contain level: ' + currentLoggingLevel ); - Integer returnedOrdinal = componentLoggerSettings.supportedLoggingLevels.get(currentLoggingLevel.name()); System.assertEquals(currentLoggingLevel.ordinal(), returnedOrdinal); } @@ -42,24 +36,18 @@ private class ComponentLogger_Tests { @IsTest static void it_should_return_aura_exception_when_it_breaks() { - Test.startTest(); try { ComponentLogger.saveComponentLogEntries(null, null); System.assert(false, 'This assert shouldn\'t run since this is a negative test'); } catch (Exception apexException) { System.assertEquals(AuraHandledException.class.getName(), apexException.getTypeName()); } - Test.stopTest(); } @IsTest static void it_should_save_component_log_entry() { - upsert LoggerSettings__c.getInstance(); - System.assertEquals(0, [SELECT COUNT() FROM Log__c]); - + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User(FirstName = UserInfo.getFirstName(), Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); - - List componentLogEntries = new List(); ComponentLogger.ComponentLogEntry componentLogEntry = new ComponentLogger.ComponentLogEntry(); componentLogEntry.loggingLevel = LoggingLevel.INFO.name(); componentLogEntry.message = 'hello, world'; @@ -67,36 +55,31 @@ private class ComponentLogger_Tests { componentLogEntry.record = currentUser; componentLogEntry.timestamp = System.now().addDays(-1 / 24); componentLogEntry.tags = new List{ 'some tag', 'one more tag' }; - componentLogEntries.add(componentLogEntry); - - Test.startTest(); - ComponentLogger.saveComponentLogEntries(componentLogEntries, null); - Test.stopTest(); - - List logEntries = [ - SELECT Id, LoggingLevel__c, Message__c, RecordId__c, RecordJson__c, RecordSObjectType__c, Timestamp__c - FROM LogEntry__c - ]; - System.assertEquals(1, logEntries.size()); - - LogEntry__c logEntry = logEntries.get(0); - - System.assertEquals(componentLogEntry.loggingLevel, logEntry.LoggingLevel__c); - System.assertEquals(componentLogEntry.message, logEntry.Message__c); - System.assertEquals(componentLogEntry.recordId, logEntry.RecordId__c); - System.assertEquals(JSON.serializePretty(currentUser), logEntry.RecordJson__c); - System.assertEquals(Schema.SObjectType.User.getName(), logEntry.RecordSObjectType__c); - System.assertEquals(componentLogEntry.timestamp, logEntry.Timestamp__c); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + ComponentLogger.saveComponentLogEntries(new List{ componentLogEntry }, null); + + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c); + System.assertEquals(componentLogEntry.loggingLevel, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(componentLogEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals(componentLogEntry.recordId, publishedLogEntryEvent.RecordId__c); + System.assertEquals(JSON.serializePretty(currentUser), publishedLogEntryEvent.RecordJson__c); + System.assertEquals(Schema.SObjectType.User.getName(), publishedLogEntryEvent.RecordSObjectType__c); + System.assertEquals(componentLogEntry.timestamp, publishedLogEntryEvent.Timestamp__c); } @IsTest static void it_should_save_component_log_entry_with_queueable_job() { - upsert LoggerSettings__c.getInstance(); - System.assertEquals(0, [SELECT COUNT() FROM Log__c]); - + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerDataStore.setMock(LoggerMockDataStore.getJobQueue()); User currentUser = new User(FirstName = UserInfo.getFirstName(), Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); - - List componentLogEntries = new List(); + System.assertEquals(0, Limits.getQueueableJobs(), 'Test should start with 0 queueable jobs used'); ComponentLogger.ComponentLogEntry componentLogEntry = new ComponentLogger.ComponentLogEntry(); componentLogEntry.loggingLevel = LoggingLevel.INFO.name(); componentLogEntry.message = 'hello, world'; @@ -104,45 +87,39 @@ private class ComponentLogger_Tests { componentLogEntry.record = currentUser; componentLogEntry.timestamp = System.now().addDays(-1 / 24); componentLogEntry.tags = new List{ 'some tag', 'one more tag' }; - componentLogEntries.add(componentLogEntry); - Test.startTest(); - System.assertEquals(0, Limits.getQueueableJobs(), 'Test should start with 0 queueable jobs used'); - - ComponentLogger.saveComponentLogEntries(componentLogEntries, Logger.SaveMethod.QUEUEABLE.name()); - - System.assertEquals(1, Limits.getQueueableJobs(), 'Log entries should have been saved using the QUEUEABLE save method'); - Test.stopTest(); - - List logEntries = [ - SELECT Id, LoggingLevel__c, Message__c, RecordId__c, RecordJson__c, RecordSObjectType__c, Timestamp__c - FROM LogEntry__c - ]; - System.assertEquals(1, logEntries.size()); - - LogEntry__c logEntry = logEntries.get(0); - - System.assertEquals(componentLogEntry.loggingLevel, logEntry.LoggingLevel__c); - System.assertEquals(componentLogEntry.message, logEntry.Message__c); - System.assertEquals(componentLogEntry.recordId, logEntry.RecordId__c); - System.assertEquals(JSON.serializePretty(currentUser), logEntry.RecordJson__c); - System.assertEquals(Schema.SObjectType.User.getName(), logEntry.RecordSObjectType__c); - System.assertEquals(componentLogEntry.timestamp, logEntry.Timestamp__c); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + System.assertEquals(0, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + + ComponentLogger.saveComponentLogEntries(new List{ componentLogEntry }, Logger.SaveMethod.QUEUEABLE.name()); + System.assertEquals(1, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + LoggerMockDataStore.getJobQueue().executeJobs(); + + System.assertEquals(Logger.SaveMethod.QUEUEABLE.name(), Logger.lastSaveMethodNameUsed); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c); + System.assertEquals(componentLogEntry.loggingLevel, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(componentLogEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals(componentLogEntry.recordId, publishedLogEntryEvent.RecordId__c); + System.assertEquals(JSON.serializePretty(currentUser), publishedLogEntryEvent.RecordJson__c); + System.assertEquals(Schema.SObjectType.User.getName(), publishedLogEntryEvent.RecordSObjectType__c); + System.assertEquals(componentLogEntry.timestamp, publishedLogEntryEvent.Timestamp__c); } @IsTest static void it_should_save_component_log_entry_with_javascript_error() { - upsert LoggerSettings__c.getInstance(); - System.assertEquals(0, [SELECT COUNT() FROM Log__c]); - + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User(FirstName = UserInfo.getFirstName(), Id = UserInfo.getUserId(), ProfileId = UserInfo.getProfileId()); - - List componentLogEntries = new List(); ComponentLogger.ComponentError mockComponentError = new ComponentLogger.ComponentError(); mockComponentError.message = 'some javascript error message'; mockComponentError.stack = 'some \nstack \ntrace \nstring'; mockComponentError.type = 'ReferenceError'; - ComponentLogger.ComponentLogEntry componentLogEntry = new ComponentLogger.ComponentLogEntry(); componentLogEntry.error = mockComponentError; componentLogEntry.loggingLevel = LoggingLevel.INFO.name(); @@ -151,137 +128,112 @@ private class ComponentLogger_Tests { componentLogEntry.record = currentUser; componentLogEntry.timestamp = System.now().addDays(-1 / 24); componentLogEntry.tags = new List{ 'some tag', 'one more tag' }; - componentLogEntries.add(componentLogEntry); - - Test.startTest(); - ComponentLogger.saveComponentLogEntries(componentLogEntries, null); - Test.stopTest(); - - List logEntries = [ - SELECT Id, LoggingLevel__c, Message__c, ExceptionMessage__c, ExceptionStackTrace__c, ExceptionType__c, Timestamp__c - FROM LogEntry__c - ]; - System.assertEquals(1, logEntries.size()); - - LogEntry__c logEntry = logEntries.get(0); - - System.assertEquals(componentLogEntry.loggingLevel, logEntry.LoggingLevel__c); - System.assertEquals(componentLogEntry.message, logEntry.Message__c); - System.assertEquals(componentLogEntry.timestamp, logEntry.Timestamp__c); - System.assertEquals(componentLogEntry.error.message, logEntry.ExceptionMessage__c); - System.assertEquals(componentLogEntry.error.stack, logEntry.ExceptionStackTrace__c); - System.assertEquals(ComponentLogger.EXCEPTION_TYPE_PREFIX + componentLogEntry.error.type, logEntry.ExceptionType__c); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + ComponentLogger.saveComponentLogEntries(new List{ componentLogEntry }, null); + + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c); + System.assertEquals(componentLogEntry.loggingLevel, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(componentLogEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals(componentLogEntry.timestamp, publishedLogEntryEvent.Timestamp__c); + System.assertEquals(componentLogEntry.error.message, publishedLogEntryEvent.ExceptionMessage__c); + System.assertEquals(componentLogEntry.error.stack, publishedLogEntryEvent.ExceptionStackTrace__c); + System.assertEquals(ComponentLogger.EXCEPTION_TYPE_PREFIX + componentLogEntry.error.type, publishedLogEntryEvent.ExceptionType__c); } @IsTest static void it_should_set_log_scenario() { - System.assertEquals(0, [SELECT COUNT() FROM Log__c]); + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); Logger.getUserSettings().LoggingLevel__c = LoggingLevel.FINEST.name(); - - List componentLogEntries = new List(); ComponentLogger.ComponentLogEntry componentLogEntry = new ComponentLogger.ComponentLogEntry(); componentLogEntry.loggingLevel = LoggingLevel.INFO.name(); componentLogEntry.message = 'hello, world'; componentLogEntry.scenario = 'Some scenario'; componentLogEntry.timestamp = System.now(); - componentLogEntries.add(componentLogEntry); - - Test.startTest(); - ComponentLogger.saveComponentLogEntries(componentLogEntries, null); - Test.stopTest(); - - Log__c log = [SELECT Id, Scenario__c FROM Log__c]; - System.assertEquals(componentLogEntry.scenario, log.Scenario__c); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + ComponentLogger.saveComponentLogEntries(new List{ componentLogEntry }, null); + + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c); + System.assertEquals(componentLogEntry.scenario, publishedLogEntryEvent.Scenario__c); } @IsTest static void it_should_parse_aura_component_stack_trace() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); String expectedComponentType = 'Aura'; String expectedComponentApiName = 'c/loggerAuraDemo'; String expectedComponentFunctionName = 'saveLogAuraExample'; - - List componentLogEntries = new List(); ComponentLogger.ComponentLogEntry componentLogEntry = new ComponentLogger.ComponentLogEntry(); componentLogEntry.loggingLevel = LoggingLevel.INFO.name(); componentLogEntry.message = 'hello, world'; componentLogEntry.stack = getMockAuraComponentStackTrace(); componentLogEntry.timestamp = System.now().addDays(-1 / 24); - componentLogEntries.add(componentLogEntry); - - Test.startTest(); - ComponentLogger.saveComponentLogEntries(componentLogEntries, null); - Test.stopTest(); - - List logEntries = [ - SELECT - Id, - LoggingLevel__c, - Message__c, - ComponentApiName__c, - ComponentFunctionName__c, - ComponentType__c, - OriginLocation__c, - OriginType__c, - Timestamp__c - FROM LogEntry__c - ]; - System.assertEquals(1, logEntries.size()); - - LogEntry__c logEntry = logEntries.get(0); - - System.assertEquals(componentLogEntry.loggingLevel, logEntry.LoggingLevel__c, logEntry); - System.assertEquals(componentLogEntry.message, logEntry.Message__c, logEntry); - System.assertEquals('Component', logEntry.OriginType__c, logEntry); - System.assertEquals(expectedComponentApiName + '.' + expectedComponentFunctionName, logEntry.OriginLocation__c, logEntry); - System.assertEquals(expectedComponentApiName, logEntry.ComponentApiName__c, logEntry); - System.assertEquals(expectedComponentFunctionName, logEntry.ComponentFunctionName__c, logEntry); - System.assertEquals(expectedComponentType, logEntry.ComponentType__c, logEntry); - System.assertEquals(componentLogEntry.timestamp, logEntry.Timestamp__c, logEntry); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + ComponentLogger.saveComponentLogEntries(new List{ componentLogEntry }, null); + + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c); + System.assertEquals(componentLogEntry.loggingLevel, publishedLogEntryEvent.LoggingLevel__c, publishedLogEntryEvent); + System.assertEquals(componentLogEntry.message, publishedLogEntryEvent.Message__c, publishedLogEntryEvent); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c, publishedLogEntryEvent); + System.assertEquals(expectedComponentApiName + '.' + expectedComponentFunctionName, publishedLogEntryEvent.OriginLocation__c, publishedLogEntryEvent); + // TODO Move these asserts to LogEntryHandler_Tests + // System.assertEquals(expectedComponentApiName, publishedLogEntryEvent.ComponentApiName__c, publishedLogEntryEvent); + // System.assertEquals(expectedComponentFunctionName, publishedLogEntryEvent.ComponentFunctionName__c, publishedLogEntryEvent); + System.assertEquals(expectedComponentType, publishedLogEntryEvent.ComponentType__c, publishedLogEntryEvent); + System.assertEquals(componentLogEntry.timestamp, publishedLogEntryEvent.Timestamp__c, publishedLogEntryEvent); } @IsTest static void it_should_parse_web_component_stack_trace() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); String expectedComponentType = 'LWC'; String expectedComponentApiName = 'c/loggerLWCDemo'; String expectedComponentFunctionName = 'saveLogWebExample'; - - List componentLogEntries = new List(); ComponentLogger.ComponentLogEntry componentLogEntry = new ComponentLogger.ComponentLogEntry(); componentLogEntry.loggingLevel = LoggingLevel.INFO.name(); componentLogEntry.message = 'hello, world'; componentLogEntry.stack = getMockWebComponentStackTrace(); componentLogEntry.timestamp = System.now().addDays(-1 / 24); - componentLogEntries.add(componentLogEntry); - - Test.startTest(); - ComponentLogger.saveComponentLogEntries(componentLogEntries, null); - Test.stopTest(); - - List logEntries = [ - SELECT - Id, - LoggingLevel__c, - Message__c, - ComponentApiName__c, - ComponentFunctionName__c, - ComponentType__c, - OriginLocation__c, - OriginType__c, - Timestamp__c - FROM LogEntry__c - ]; - System.assertEquals(1, logEntries.size()); - - LogEntry__c logEntry = logEntries.get(0); - - System.assertEquals(componentLogEntry.loggingLevel, logEntry.LoggingLevel__c, logEntry); - System.assertEquals(componentLogEntry.message, logEntry.Message__c, logEntry); - System.assertEquals('Component', logEntry.OriginType__c, logEntry); - System.assertEquals(expectedComponentApiName + '.' + expectedComponentFunctionName, logEntry.OriginLocation__c, logEntry); - System.assertEquals(expectedComponentApiName, logEntry.ComponentApiName__c, logEntry); - System.assertEquals(expectedComponentFunctionName, logEntry.ComponentFunctionName__c, logEntry); - System.assertEquals(expectedComponentType, logEntry.ComponentType__c, logEntry); - System.assertEquals(componentLogEntry.timestamp, logEntry.Timestamp__c, logEntry); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + ComponentLogger.saveComponentLogEntries(new List{ componentLogEntry }, null); + + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c); + System.assertEquals(componentLogEntry.loggingLevel, publishedLogEntryEvent.LoggingLevel__c, publishedLogEntryEvent); + System.assertEquals(componentLogEntry.message, publishedLogEntryEvent.Message__c, publishedLogEntryEvent); + System.assertEquals('Component', publishedLogEntryEvent.OriginType__c, publishedLogEntryEvent); + System.assertEquals(expectedComponentApiName + '.' + expectedComponentFunctionName, publishedLogEntryEvent.OriginLocation__c, publishedLogEntryEvent); + // TODO Move these asserts to LogEntryHandler_Tests + // System.assertEquals(expectedComponentApiName, publishedLogEntryEvent.ComponentApiName__c, publishedLogEntryEvent); + // System.assertEquals(expectedComponentFunctionName, publishedLogEntryEvent.ComponentFunctionName__c, publishedLogEntryEvent); + System.assertEquals(expectedComponentType, publishedLogEntryEvent.ComponentType__c, publishedLogEntryEvent); + System.assertEquals(componentLogEntry.timestamp, publishedLogEntryEvent.Timestamp__c, publishedLogEntryEvent); } private static String getMockAuraComponentStackTrace() { diff --git a/nebula-logger/core/tests/logger-engine/classes/FlowCollectionLogEntry_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/FlowCollectionLogEntry_Tests.cls index f2b4edcf7..a7906acc7 100644 --- a/nebula-logger/core/tests/logger-engine/classes/FlowCollectionLogEntry_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/FlowCollectionLogEntry_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class FlowCollectionLogEntry_Tests { static FlowCollectionLogEntry createFlowCollectionLogEntry() { FlowCollectionLogEntry flowCollectionEntry = new FlowCollectionLogEntry(); @@ -17,311 +17,276 @@ private class FlowCollectionLogEntry_Tests { @IsTest static void it_should_save_entry_when_logging_level_met() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowCollectionEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowCollectionEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); flowCollectionEntry.loggingLevelName = flowCollectionEntryLoggingLevel.name(); flowCollectionEntry.records = new List{ currentUser }; - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); String expectedUserJson = JSON.serializePretty(new List{ currentUser }); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordCollectionType__c, RecordJson__c, RecordSObjectType__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowCollectionEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowCollectionEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - System.assertEquals(null, logEntry.RecordId__c); - System.assertEquals('List', logEntry.RecordCollectionType__c); - System.assertEquals('User', logEntry.RecordSObjectType__c); - System.assertEquals(expectedUserJson, logEntry.RecordJson__c); + System.assertEquals(flowCollectionEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowCollectionEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(null, publishedLogEntryEvent.RecordId__c); + System.assertEquals('List', publishedLogEntryEvent.RecordCollectionType__c); + System.assertEquals('User', publishedLogEntryEvent.RecordSObjectType__c); + System.assertEquals(expectedUserJson, publishedLogEntryEvent.RecordJson__c); } @IsTest static void it_should_auto_save_entry_when_saveLog_is_true() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowCollectionEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowCollectionEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); flowCollectionEntry.loggingLevelName = flowCollectionEntryLoggingLevel.name(); flowCollectionEntry.records = new List{ currentUser }; flowCollectionEntry.saveLog = true; - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); - Test.getEventBus().deliver(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); - Test.stopTest(); + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); String expectedUserJson = JSON.serializePretty(new List{ currentUser }); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordCollectionType__c, RecordJson__c, RecordSObjectType__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowCollectionEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowCollectionEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - System.assertEquals(null, logEntry.RecordId__c); - System.assertEquals('List', logEntry.RecordCollectionType__c); - System.assertEquals('User', logEntry.RecordSObjectType__c); - System.assertEquals(expectedUserJson, logEntry.RecordJson__c); + System.assertEquals(flowCollectionEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowCollectionEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(null, publishedLogEntryEvent.RecordId__c); + System.assertEquals('List', publishedLogEntryEvent.RecordCollectionType__c); + System.assertEquals('User', publishedLogEntryEvent.RecordSObjectType__c); + System.assertEquals(expectedUserJson, publishedLogEntryEvent.RecordJson__c); } @IsTest static void it_should_auto_save_entry_with_save_method_when_saveMethodName_specified() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerDataStore.setMock(LoggerMockDataStore.getJobQueue()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowCollectionEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowCollectionEntryLoggingLevel.ordinal()); - Test.startTest(); System.assertEquals(0, Limits.getQueueableJobs(), 'Test should start with 0 queueable jobs used'); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); flowCollectionEntry.loggingLevelName = flowCollectionEntryLoggingLevel.name(); flowCollectionEntry.records = new List{ currentUser }; flowCollectionEntry.saveLog = true; flowCollectionEntry.saveMethodName = Logger.SaveMethod.QUEUEABLE.name(); - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); - Test.getEventBus().deliver(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + System.assertEquals(0, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); - System.assertEquals(1, Limits.getQueueableJobs(), 'Log entries should have been saved using the QUEUEABLE save method'); - Test.stopTest(); + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); + System.assertEquals(1, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + LoggerMockDataStore.getJobQueue().executeJobs(); + System.assertEquals(Logger.SaveMethod.QUEUEABLE.name(), Logger.lastSaveMethodNameUsed); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); String expectedUserJson = JSON.serializePretty(new List{ currentUser }); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordCollectionType__c, RecordJson__c, RecordSObjectType__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowCollectionEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowCollectionEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - System.assertEquals(null, logEntry.RecordId__c); - System.assertEquals('List', logEntry.RecordCollectionType__c); - System.assertEquals('User', logEntry.RecordSObjectType__c); - System.assertEquals(expectedUserJson, logEntry.RecordJson__c); + System.assertEquals(flowCollectionEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowCollectionEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(null, publishedLogEntryEvent.RecordId__c); + System.assertEquals('List', publishedLogEntryEvent.RecordCollectionType__c); + System.assertEquals('User', publishedLogEntryEvent.RecordSObjectType__c); + System.assertEquals(expectedUserJson, publishedLogEntryEvent.RecordJson__c); } @IsTest static void it_should_not_save_entry_when_logging_level_not_met() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.ERROR; LoggingLevel flowCollectionEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() > flowCollectionEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); flowCollectionEntry.loggingLevelName = flowCollectionEntryLoggingLevel.name(); flowCollectionEntry.records = new List{ currentUser }; - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); - Test.getEventBus().deliver(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); System.assertEquals(0, Logger.getBufferSize()); + Logger.saveLog(); - Test.stopTest(); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); } @IsTest static void it_should_use_debug_as_default_level_when_faultMessage_is_null() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel expectedEntryLoggingLevel = LoggingLevel.DEBUG; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = expectedEntryLoggingLevel.name(); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); + System.assertEquals(null, flowCollectionEntry.faultMessage); + System.assertEquals(null, flowCollectionEntry.loggingLevelName); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); - FlowCollectionLogEntry flowEntry = createFlowCollectionLogEntry(); - System.assertEquals(null, flowEntry.faultMessage); - System.assertEquals(null, flowEntry.loggingLevelName); - - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowEntry }); - + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - LogEntry__c logEntry = [ - SELECT - Id, - ExceptionMessage__c, - ExceptionType__c, - LoggingLevel__c, - Message__c, - OriginType__c, - OriginLocation__c, - RecordId__c, - RecordCollectionType__c, - RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(null, logEntry.ExceptionMessage__c); - System.assertEquals(null, logEntry.ExceptionType__c); - System.assertEquals(expectedEntryLoggingLevel.name(), logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(null, publishedLogEntryEvent.ExceptionMessage__c); + System.assertEquals(null, publishedLogEntryEvent.ExceptionType__c); + System.assertEquals(expectedEntryLoggingLevel.name(), publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowCollectionEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_use_error_as_default_level_when_faultMessage_is_not_null() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel expectedEntryLoggingLevel = LoggingLevel.ERROR; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = LoggingLevel.FINEST.name(); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); + flowCollectionEntry.faultMessage = 'Whoops, a Flow error has occurred.'; + System.assertEquals(null, flowCollectionEntry.loggingLevelName); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); - FlowCollectionLogEntry flowEntry = createFlowCollectionLogEntry(); - flowEntry.faultMessage = 'Whoops, a Flow error has occurred.'; - System.assertEquals(null, flowEntry.loggingLevelName); - - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowEntry }); - + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - LogEntry__c logEntry = [ - SELECT - Id, - ExceptionMessage__c, - ExceptionType__c, - LoggingLevel__c, - Message__c, - OriginType__c, - OriginLocation__c, - RecordId__c, - RecordCollectionType__c, - RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowEntry.faultMessage, logEntry.ExceptionMessage__c); - System.assertEquals('Flow.FaultError', logEntry.ExceptionType__c); - System.assertEquals(expectedEntryLoggingLevel.name(), logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowCollectionEntry.faultMessage, publishedLogEntryEvent.ExceptionMessage__c); + System.assertEquals('Flow.FaultError', publishedLogEntryEvent.ExceptionType__c); + System.assertEquals(expectedEntryLoggingLevel.name(), publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowCollectionEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_set_log_scenario() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - FlowCollectionLogEntry flowEntry = createFlowCollectionLogEntry(); - flowEntry.loggingLevelName = userLoggingLevel.name(); - flowEntry.scenario = 'Some scenario'; - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowEntry }); - System.assertEquals(1, Logger.getBufferSize()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); + flowCollectionEntry.loggingLevelName = userLoggingLevel.name(); + flowCollectionEntry.scenario = 'Some scenario'; + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); + System.assertEquals(1, Logger.getBufferSize()); Logger.saveLog(); - Test.getEventBus().deliver(); - String transactionId = Logger.getTransactionId(); - - Test.stopTest(); - Log__c log = [SELECT Id, Scenario__c FROM Log__c WHERE TransactionId__c = :transactionId]; - System.assertEquals(flowEntry.scenario, log.Scenario__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowCollectionEntry.scenario, publishedLogEntryEvent.Scenario__c); } @IsTest static void it_should_add_tags_to_log_entry() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowCollectionEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowCollectionEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); List tags = new List{ 'first tag', 'SECOND TAG' }; FlowCollectionLogEntry flowCollectionEntry = createFlowCollectionLogEntry(); flowCollectionEntry.loggingLevelName = flowCollectionEntryLoggingLevel.name(); flowCollectionEntry.tagsString = String.join(tags, ', '); - FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowCollectionLogEntry.addFlowCollectionEntries(new List{ flowCollectionEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - LogEntry__c logEntry = [SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c FROM LogEntry__c ORDER BY CreatedDate LIMIT 1]; - System.assertEquals(flowCollectionEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowCollectionEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - - List logEntryTags = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; - System.assertEquals(tags.size(), logEntryTags.size(), logEntryTags); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowCollectionEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowCollectionEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + List publishedLogEntryEventTags = publishedLogEntryEvent.Tags__c.split('\n'); + System.assertEquals(tags.size(), publishedLogEntryEventTags.size(), publishedLogEntryEventTags); Set tagsSet = new Set(tags); - for (LogEntryTag__c logEntryTag : logEntryTags) { - System.assert(tagsSet.contains(logEntryTag.Tag__r.Name)); + for (String publishedTag : publishedLogEntryEventTags) { + publishedTag = publishedTag.trim(); + System.assert(tagsSet.contains(publishedTag), publishedTag + ' not found in expected tags set: ' + tagsSet); } } } diff --git a/nebula-logger/core/tests/logger-engine/classes/FlowLogEntry_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/FlowLogEntry_Tests.cls index ea0d642b7..984b5dea2 100644 --- a/nebula-logger/core/tests/logger-engine/classes/FlowLogEntry_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/FlowLogEntry_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class FlowLogEntry_Tests { static FlowLogEntry createFlowLogEntry() { FlowLogEntry flowEntry = new FlowLogEntry(); @@ -17,236 +17,234 @@ private class FlowLogEntry_Tests { @IsTest static void it_should_save_entry_when_logging_level_met() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowLogEntry flowEntry = createFlowLogEntry(); flowEntry.loggingLevelName = flowEntryLoggingLevel.name(); - FlowLogEntry.addFlowEntries(new List{ flowEntry }); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowLogEntry.addFlowEntries(new List{ flowEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - LogEntry__c logEntry = [SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c FROM LogEntry__c ORDER BY CreatedDate LIMIT 1]; - System.assertEquals(flowEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_auto_save_entry_when_saveLog_is_true() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowLogEntry flowEntry = createFlowLogEntry(); flowEntry.loggingLevelName = flowEntryLoggingLevel.name(); flowEntry.saveLog = true; + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowLogEntry.addFlowEntries(new List{ flowEntry }); - Test.getEventBus().deliver(); - - Test.stopTest(); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_auto_save_entry_with_save_method_when_saveMethodName_specified() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerDataStore.setMock(LoggerMockDataStore.getJobQueue()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowEntryLoggingLevel.ordinal()); - - Test.startTest(); - System.assertEquals(0, Limits.getQueueableJobs(), 'Test should start with 0 queueable jobs used'); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowLogEntry flowEntry = createFlowLogEntry(); flowEntry.loggingLevelName = flowEntryLoggingLevel.name(); flowEntry.saveLog = true; flowEntry.saveMethodName = Logger.SaveMethod.QUEUEABLE.name(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + System.assertEquals(0, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + FlowLogEntry.addFlowEntries(new List{ flowEntry }); - Test.getEventBus().deliver(); - - System.assertEquals(1, Limits.getQueueableJobs(), 'Log entries should have been saved using the QUEUEABLE save method'); - Test.stopTest(); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + System.assertEquals(1, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + LoggerMockDataStore.getJobQueue().executeJobs(); + + System.assertEquals(Logger.SaveMethod.QUEUEABLE.name(), Logger.lastSaveMethodNameUsed); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_not_save_entry_when_logging_level_not_met() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.ERROR; LoggingLevel flowEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() > flowEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowLogEntry flowEntry = createFlowLogEntry(); flowEntry.loggingLevelName = flowEntryLoggingLevel.name(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowLogEntry.addFlowEntries(new List{ flowEntry }); - Test.getEventBus().deliver(); + Logger.saveLog(); System.assertEquals(0, Logger.getBufferSize()); - - Test.stopTest(); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); } @IsTest static void it_should_use_debug_as_default_level_when_faultMessage_is_null() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel expectedEntryLoggingLevel = LoggingLevel.DEBUG; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = expectedEntryLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowLogEntry flowEntry = createFlowLogEntry(); System.assertEquals(null, flowEntry.faultMessage); System.assertEquals(null, flowEntry.loggingLevelName); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); FlowLogEntry.addFlowEntries(new List{ flowEntry }); - System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - - LogEntry__c logEntry = [ - SELECT Id, ExceptionMessage__c, ExceptionType__c, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(null, logEntry.ExceptionMessage__c); - System.assertEquals(null, logEntry.ExceptionType__c); - System.assertEquals(expectedEntryLoggingLevel.name(), logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(null, publishedLogEntryEvent.ExceptionMessage__c); + System.assertEquals(null, publishedLogEntryEvent.ExceptionType__c); + System.assertEquals(expectedEntryLoggingLevel.name(), publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_use_error_as_default_level_when_faultMessage_is_not_null() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel expectedEntryLoggingLevel = LoggingLevel.ERROR; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = LoggingLevel.FINEST.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowLogEntry flowEntry = createFlowLogEntry(); flowEntry.faultMessage = 'Whoops, a Flow error has occurred.'; System.assertEquals(null, flowEntry.loggingLevelName); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); FlowLogEntry.addFlowEntries(new List{ flowEntry }); - System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - - LogEntry__c logEntry = [ - SELECT Id, ExceptionMessage__c, ExceptionType__c, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowEntry.faultMessage, logEntry.ExceptionMessage__c); - System.assertEquals('Flow.FaultError', logEntry.ExceptionType__c); - System.assertEquals(expectedEntryLoggingLevel.name(), logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.faultMessage, publishedLogEntryEvent.ExceptionMessage__c); + System.assertEquals('Flow.FaultError', publishedLogEntryEvent.ExceptionType__c); + System.assertEquals(expectedEntryLoggingLevel.name(), publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_set_log_scenario() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowLogEntry flowEntry = createFlowLogEntry(); flowEntry.loggingLevelName = userLoggingLevel.name(); flowEntry.scenario = 'Some scenario'; + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowLogEntry.addFlowEntries(new List{ flowEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - String transactionId = Logger.getTransactionId(); - Test.stopTest(); - - Log__c log = [SELECT Id, Scenario__c FROM Log__c WHERE TransactionId__c = :transactionId]; - System.assertEquals(flowEntry.scenario, log.Scenario__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.scenario, publishedLogEntryEvent.Scenario__c); } @IsTest static void it_should_add_tags_to_log_entry() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowEntryLoggingLevel.ordinal()); - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); List tags = new List{ 'first tag', 'SECOND TAG' }; FlowLogEntry flowEntry = createFlowLogEntry(); flowEntry.loggingLevelName = flowEntryLoggingLevel.name(); flowEntry.tagsString = String.join(tags, ', '); - FlowLogEntry.addFlowEntries(new List{ flowEntry }); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowLogEntry.addFlowEntries(new List{ flowEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - Test.stopTest(); - - LogEntry__c logEntry = [SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c FROM LogEntry__c ORDER BY CreatedDate LIMIT 1]; - System.assertEquals(flowEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - - List logEntryTags = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; - System.assertEquals(tags.size(), logEntryTags.size(), logEntryTags); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + List publishedLogEntryEventTags = publishedLogEntryEvent.Tags__c.split('\n'); + System.assertEquals(tags.size(), publishedLogEntryEventTags.size(), publishedLogEntryEventTags); Set tagsSet = new Set(tags); - for (LogEntryTag__c logEntryTag : logEntryTags) { - System.assert(tagsSet.contains(logEntryTag.Tag__r.Name)); + for (String publishedTag : publishedLogEntryEventTags) { + publishedTag = publishedTag.trim(); + System.assert(tagsSet.contains(publishedTag), publishedTag + ' not found in expected tags set: ' + tagsSet); } } } diff --git a/nebula-logger/core/tests/logger-engine/classes/FlowLogger_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/FlowLogger_Tests.cls index 467a4a7e9..c5bdb7525 100644 --- a/nebula-logger/core/tests/logger-engine/classes/FlowLogger_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/FlowLogger_Tests.cls @@ -4,91 +4,107 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class FlowLogger_Tests { @IsTest static void it_should_add_entry_to_logger_buffer() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel entryLoggingLevel = LoggingLevel.DEBUG; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = entryLoggingLevel.name(); - - FlowLogger.LogEntry logEntry = new FlowLogger.LogEntry(); - logEntry.flowName = 'MyFlow'; - logEntry.message = 'hello from Flow'; - logEntry.loggingLevelName = entryLoggingLevel.name(); - logEntry.saveLog = false; - logEntry.timestamp = System.now(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); System.assertEquals(0, Logger.getBufferSize()); System.assertEquals(0, [SELECT COUNT() FROM LogEntry__c]); - - FlowLogger.addEntries(new List{ logEntry }); + FlowLogger.LogEntry flowEntry = new FlowLogger.LogEntry(); + flowEntry.flowName = 'MyFlow'; + flowEntry.message = 'hello from Flow'; + flowEntry.loggingLevelName = entryLoggingLevel.name(); + flowEntry.saveLog = false; + flowEntry.timestamp = System.now(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + FlowLogger.addEntries(new List{ flowEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - System.assertEquals(1, [SELECT COUNT() FROM LogEntry__c]); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(flowEntry.timestamp, publishedLogEntryEvent.Timestamp__c); } @IsTest static void it_should_auto_save_entry_when_saveLog_is_true() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel entryLoggingLevel = LoggingLevel.DEBUG; - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = entryLoggingLevel.name(); - - FlowLogger.LogEntry logEntry = new FlowLogger.LogEntry(); - logEntry.flowName = 'MyFlow'; - logEntry.message = 'hello from Flow'; - logEntry.loggingLevelName = entryLoggingLevel.name(); - logEntry.saveLog = true; - logEntry.timestamp = System.now(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); System.assertEquals(0, Logger.getBufferSize()); System.assertEquals(0, [SELECT COUNT() FROM LogEntry__c]); + FlowLogger.LogEntry flowEntry = new FlowLogger.LogEntry(); + flowEntry.flowName = 'MyFlow'; + flowEntry.message = 'hello from Flow'; + flowEntry.loggingLevelName = entryLoggingLevel.name(); + flowEntry.saveLog = true; + flowEntry.timestamp = System.now(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + FlowLogger.addEntries(new List{ flowEntry }); - FlowLogger.addEntries(new List{ logEntry }); - Test.getEventBus().deliver(); System.assertEquals(0, Logger.getBufferSize()); - - Test.stopTest(); - - System.assertEquals(1, [SELECT COUNT() FROM LogEntry__c]); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(flowEntry.timestamp, publishedLogEntryEvent.Timestamp__c); } @IsTest static void it_should_auto_save_entry_with_save_method_when_saveMethodName_specified() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerDataStore.setMock(LoggerMockDataStore.getJobQueue()); LoggingLevel entryLoggingLevel = LoggingLevel.DEBUG; - - Test.startTest(); System.assertEquals(0, Limits.getQueueableJobs(), 'Test should start with 0 queueable jobs used'); - Logger.getUserSettings().LoggingLevel__c = entryLoggingLevel.name(); - - FlowLogger.LogEntry logEntry = new FlowLogger.LogEntry(); - logEntry.flowName = 'MyFlow'; - logEntry.message = 'hello from Flow'; - logEntry.loggingLevelName = entryLoggingLevel.name(); - logEntry.saveLog = true; - logEntry.saveMethodName = Logger.SaveMethod.QUEUEABLE.name(); - logEntry.timestamp = System.now(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); System.assertEquals(0, Logger.getBufferSize()); System.assertEquals(0, [SELECT COUNT() FROM LogEntry__c]); - - FlowLogger.addEntries(new List{ logEntry }); - Test.getEventBus().deliver(); + FlowLogger.LogEntry flowEntry = new FlowLogger.LogEntry(); + flowEntry.flowName = 'MyFlow'; + flowEntry.message = 'hello from Flow'; + flowEntry.loggingLevelName = entryLoggingLevel.name(); + flowEntry.saveLog = true; + flowEntry.saveMethodName = Logger.SaveMethod.QUEUEABLE.name(); + flowEntry.timestamp = System.now(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + + FlowLogger.addEntries(new List{ flowEntry }); + System.assertEquals(1, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + LoggerMockDataStore.getJobQueue().executeJobs(); + + System.assertEquals(Logger.SaveMethod.QUEUEABLE.name(), Logger.lastSaveMethodNameUsed); System.assertEquals(0, Logger.getBufferSize()); - - System.assertEquals(1, Limits.getQueueableJobs(), 'Log entries should have been saved using the QUEUEABLE save method'); - Test.stopTest(); - - System.assertEquals(1, [SELECT COUNT() FROM LogEntry__c]); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(flowEntry.timestamp, publishedLogEntryEvent.Timestamp__c); } } diff --git a/nebula-logger/core/tests/logger-engine/classes/FlowRecordLogEntry_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/FlowRecordLogEntry_Tests.cls index b19667cf4..ca06b3418 100644 --- a/nebula-logger/core/tests/logger-engine/classes/FlowRecordLogEntry_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/FlowRecordLogEntry_Tests.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class FlowRecordLogEntry_Tests { static FlowRecordLogEntry createFlowRecordLogEntry() { FlowRecordLogEntry flowRecordEntry = new FlowRecordLogEntry(); @@ -17,285 +17,270 @@ private class FlowRecordLogEntry_Tests { @IsTest static void it_should_save_entry_when_logging_level_met() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowRecordEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowRecordEntryLoggingLevel.ordinal()); - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); flowRecordEntry.loggingLevelName = flowRecordEntryLoggingLevel.name(); flowRecordEntry.record = currentUser; - FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); String expectedUserJson = JSON.serializePretty(currentUser); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowRecordEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowRecordEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - System.assertEquals(currentUser.Id, logEntry.RecordId__c); - System.assertEquals(expectedUserJson, logEntry.RecordJson__c); + System.assertEquals(flowRecordEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowRecordEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(currentUser.Id, publishedLogEntryEvent.RecordId__c); + System.assertEquals(expectedUserJson, publishedLogEntryEvent.RecordJson__c); } @IsTest static void it_should_auto_save_entry_with_save_method_when_saveMethodName_specified() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); + LoggerDataStore.setMock(LoggerMockDataStore.getJobQueue()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowRecordEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowRecordEntryLoggingLevel.ordinal()); - - Test.startTest(); System.assertEquals(0, Limits.getQueueableJobs(), 'Test should start with 0 queueable jobs used'); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); flowRecordEntry.loggingLevelName = flowRecordEntryLoggingLevel.name(); flowRecordEntry.record = currentUser; flowRecordEntry.saveLog = true; flowRecordEntry.saveMethodName = Logger.SaveMethod.QUEUEABLE.name(); - FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); - Test.getEventBus().deliver(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + System.assertEquals(0, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); - System.assertEquals(1, Limits.getQueueableJobs(), 'Log entries should have been saved using the QUEUEABLE save method'); - Test.stopTest(); + FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); + System.assertEquals(1, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + LoggerMockDataStore.getJobQueue().executeJobs(); + System.assertEquals(Logger.SaveMethod.QUEUEABLE.name(), Logger.lastSaveMethodNameUsed); + System.assertEquals(1, LoggerMockDataStore.getJobQueue().getEnqueuedJobs().size()); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); String expectedUserJson = JSON.serializePretty(currentUser); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowRecordEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowRecordEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - System.assertEquals(currentUser.Id, logEntry.RecordId__c); - System.assertEquals(expectedUserJson, logEntry.RecordJson__c); + System.assertEquals(flowRecordEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowRecordEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(currentUser.Id, publishedLogEntryEvent.RecordId__c); + System.assertEquals(expectedUserJson, publishedLogEntryEvent.RecordJson__c); } @IsTest static void it_should_auto_save_entry_when_saveLog_is_true() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowRecordEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowRecordEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); flowRecordEntry.loggingLevelName = flowRecordEntryLoggingLevel.name(); flowRecordEntry.record = currentUser; flowRecordEntry.saveLog = true; - FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); - Test.getEventBus().deliver(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); - Test.stopTest(); + FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); String expectedUserJson = JSON.serializePretty(currentUser); - - LogEntry__c logEntry = [ - SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowRecordEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowRecordEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - System.assertEquals(currentUser.Id, logEntry.RecordId__c); - System.assertEquals(expectedUserJson, logEntry.RecordJson__c); + System.assertEquals(flowRecordEntry.loggingLevelName, publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowRecordEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); + System.assertEquals(currentUser.Id, publishedLogEntryEvent.RecordId__c); + System.assertEquals(expectedUserJson, publishedLogEntryEvent.RecordJson__c); } @IsTest static void it_should_not_save_entry_when_logging_level_not_met() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); User currentUser = new User( Id = UserInfo.getUserId(), FirstName = UserInfo.getFirstName(), LastName = UserInfo.getLastName(), Username = UserInfo.getUserName() ); - LoggingLevel userLoggingLevel = LoggingLevel.ERROR; LoggingLevel flowRecordEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() > flowRecordEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); flowRecordEntry.loggingLevelName = flowRecordEntryLoggingLevel.name(); flowRecordEntry.record = currentUser; - FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); - Test.getEventBus().deliver(); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); System.assertEquals(0, Logger.getBufferSize()); + Logger.saveLog(); - Test.stopTest(); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); } @IsTest static void it_should_use_debug_as_default_level_when_faultMessage_is_null() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel expectedEntryLoggingLevel = LoggingLevel.DEBUG; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = expectedEntryLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); System.assertEquals(null, flowRecordEntry.faultMessage); System.assertEquals(null, flowRecordEntry.loggingLevelName); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); - System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - LogEntry__c logEntry = [ - SELECT Id, ExceptionMessage__c, ExceptionType__c, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(null, logEntry.ExceptionMessage__c); - System.assertEquals(null, logEntry.ExceptionType__c); - System.assertEquals(expectedEntryLoggingLevel.name(), logEntry.LoggingLevel__c); - System.assertEquals(flowRecordEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(null, publishedLogEntryEvent.ExceptionMessage__c); + System.assertEquals(null, publishedLogEntryEvent.ExceptionType__c); + System.assertEquals(expectedEntryLoggingLevel.name(), publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowRecordEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_use_error_as_default_level_when_faultMessage_is_not_null() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel expectedEntryLoggingLevel = LoggingLevel.ERROR; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = LoggingLevel.FINEST.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); flowRecordEntry.faultMessage = 'Whoops, a Flow error has occurred.'; System.assertEquals(null, flowRecordEntry.loggingLevelName); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); - System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - Test.stopTest(); - - LogEntry__c logEntry = [ - SELECT Id, ExceptionMessage__c, ExceptionType__c, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c, RecordId__c, RecordJson__c - FROM LogEntry__c - ORDER BY CreatedDate - LIMIT 1 - ]; - System.assertEquals(flowRecordEntry.faultMessage, logEntry.ExceptionMessage__c); - System.assertEquals('Flow.FaultError', logEntry.ExceptionType__c); - System.assertEquals(expectedEntryLoggingLevel.name(), logEntry.LoggingLevel__c); - System.assertEquals(flowRecordEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowRecordEntry.faultMessage, publishedLogEntryEvent.ExceptionMessage__c); + System.assertEquals('Flow.FaultError', publishedLogEntryEvent.ExceptionType__c); + System.assertEquals(expectedEntryLoggingLevel.name(), publishedLogEntryEvent.LoggingLevel__c); + System.assertEquals(flowRecordEntry.message, publishedLogEntryEvent.Message__c); + System.assertEquals('Flow', publishedLogEntryEvent.OriginType__c); } @IsTest static void it_should_set_log_scenario() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - FlowRecordLogEntry flowEntry = createFlowRecordLogEntry(); - flowEntry.loggingLevelName = userLoggingLevel.name(); - flowEntry.scenario = 'Some scenario'; - FlowRecordLogEntry.addFlowRecordEntries(new List{ flowEntry }); - System.assertEquals(1, Logger.getBufferSize()); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); + flowRecordEntry.loggingLevelName = userLoggingLevel.name(); + flowRecordEntry.scenario = 'Some scenario'; + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); + System.assertEquals(1, Logger.getBufferSize()); Logger.saveLog(); - Test.getEventBus().deliver(); - String transactionId = Logger.getTransactionId(); - Test.stopTest(); - - Log__c log = [SELECT Id, Scenario__c FROM Log__c WHERE TransactionId__c = :transactionId]; - System.assertEquals(flowEntry.scenario, log.Scenario__c); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + System.assertEquals(flowRecordEntry.scenario, publishedLogEntryEvent.Scenario__c); } @IsTest static void it_should_add_tags_to_log_entry() { + LoggerDataStore.setMock(LoggerMockDataStore.getEventBus()); LoggingLevel userLoggingLevel = LoggingLevel.FINEST; LoggingLevel flowRecordEntryLoggingLevel = LoggingLevel.DEBUG; System.assert(userLoggingLevel.ordinal() < flowRecordEntryLoggingLevel.ordinal()); - - Test.startTest(); - Logger.getUserSettings().LoggingLevel__c = userLoggingLevel.name(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); List tags = new List{ 'first tag', 'SECOND TAG' }; - FlowRecordLogEntry flowEntry = createFlowRecordLogEntry(); - flowEntry.loggingLevelName = flowRecordEntryLoggingLevel.name(); - flowEntry.tagsString = String.join(tags, ', '); - FlowRecordLogEntry.addFlowRecordEntries(new List{ flowEntry }); + FlowRecordLogEntry flowRecordEntry = createFlowRecordLogEntry(); + flowRecordEntry.loggingLevelName = flowRecordEntryLoggingLevel.name(); + flowRecordEntry.tagsString = String.join(tags, ', '); + System.assertEquals(0, Logger.saveLogCallCount); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(0, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + FlowRecordLogEntry.addFlowRecordEntries(new List{ flowRecordEntry }); System.assertEquals(1, Logger.getBufferSize()); - Logger.saveLog(); - Test.getEventBus().deliver(); - - Test.stopTest(); - LogEntry__c logEntry = [SELECT Id, LoggingLevel__c, Message__c, OriginType__c, OriginLocation__c FROM LogEntry__c ORDER BY CreatedDate LIMIT 1]; - System.assertEquals(flowEntry.loggingLevelName, logEntry.LoggingLevel__c); - System.assertEquals(flowEntry.message, logEntry.Message__c); - System.assertEquals('Flow', logEntry.OriginType__c); - - List logEntryTags = [SELECT Id, LogEntry__c, Tag__c, Tag__r.Name FROM LogEntryTag__c WHERE LogEntry__c = :logEntry.Id]; - System.assertEquals(tags.size(), logEntryTags.size(), logEntryTags); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(1, Logger.saveLogCallCount); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishCallCount()); + System.assertEquals(1, LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().size()); + LogEntryEvent__e publishedLogEntryEvent = (LogEntryEvent__e) LoggerMockDataStore.getEventBus().getPublishedPlatformEvents().get(0); + List publishedLogEntryEventTags = publishedLogEntryEvent.Tags__c.split('\n'); + System.assertEquals(tags.size(), publishedLogEntryEventTags.size(), publishedLogEntryEventTags); Set tagsSet = new Set(tags); - for (LogEntryTag__c logEntryTag : logEntryTags) { - System.assert(tagsSet.contains(logEntryTag.Tag__r.Name)); + for (String publishedTag : publishedLogEntryEventTags) { + publishedTag = publishedTag.trim(); + System.assert(tagsSet.contains(publishedTag), publishedTag + ' not found in expected tags set: ' + tagsSet); } } } diff --git a/nebula-logger/core/tests/logger-engine/classes/LogEntryEventBuilder_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/LogEntryEventBuilder_Tests.cls index 5c6f0e412..7b230d3c9 100644 --- a/nebula-logger/core/tests/logger-engine/classes/LogEntryEventBuilder_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/LogEntryEventBuilder_Tests.cls @@ -6,26 +6,8 @@ @SuppressWarnings( 'PMD.ApexDoc, PMD.ApexAssertionsShouldIncludeMessage, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount' ) -@IsTest +@IsTest(IsParallel=false) private class LogEntryEventBuilder_Tests { - static String getMessage() { - return 'Hello, world'; - } - - static LogMessage getLogMessage() { - return new LogMessage('The current date is {0}', System.today()); - } - - static LogEntryDataMaskRule__mdt getSocialSecurityNumberDataMaskRule() { - return new LogEntryDataMaskRule__mdt( - IsEnabled__c = true, - ApplyToMessage__c = true, - ApplyToRecordJson__c = true, - SensitiveDataRegEx__c = '(\\d{3})[- ]*(\\d{2})[- ]*(\\d{4})', - ReplacementRegEx__c = 'XXX-XX-$3' - ); - } - @IsTest static void it_should_short_circuit_when_disabled() { LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.DEBUG, false); @@ -136,48 +118,75 @@ private class LogEntryEventBuilder_Tests { } @IsTest - static void it_should_apply_data_mask_rule_to_message_when_enabled() { + static void it_should_not_apply_data_mask_rule_when_rule_disabled() { Logger.getUserSettings().IsDataMaskingEnabled__c = true; LogEntryDataMaskRule__mdt rule = getSocialSecurityNumberDataMaskRule(); - rule.ApplyToMessage__c = true; - rule.ApplyToRecordJson__c = false; - LogEntryEventBuilder.addMockDataMaskRule(rule); - + rule.IsEnabled__c = false; + LogEntryEventBuilder.setMockDataMaskRule(rule); String message = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; - String expectedSanitizedMessage = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; Account account = new Account(Name = message); String accountJson = JSON.serializePretty(account); - String expectedSanitizedAccountJson = JSON.serializePretty(new Account(Name = message)); + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setMessage(message).setRecord(account); - System.assertEquals(true, builder.getLogEntryEvent().MessageMasked__c); - System.assertNotEquals(message, builder.getLogEntryEvent().Message__c); - System.assertEquals(expectedSanitizedMessage, builder.getLogEntryEvent().Message__c); + System.assertEquals(false, builder.getLogEntryEvent().MessageMasked__c, builder.getLogEntryEvent().Message__c); + System.assertEquals(message, builder.getLogEntryEvent().Message__c); System.assertEquals(false, builder.getLogEntryEvent().RecordJsonMasked__c); System.assertEquals(accountJson, builder.getLogEntryEvent().RecordJson__c); - System.assertEquals(expectedSanitizedAccountJson, builder.getLogEntryEvent().RecordJson__c); } @IsTest - static void it_should_apply_data_mask_rule_to_record_json_when_enabled() { - Logger.getUserSettings().IsDataMaskingEnabled__c = true; + static void it_should_not_apply_data_mask_rule_when_disabled_for_user() { + Logger.getUserSettings().IsDataMaskingEnabled__c = false; LogEntryDataMaskRule__mdt rule = getSocialSecurityNumberDataMaskRule(); - rule.ApplyToMessage__c = false; - rule.ApplyToRecordJson__c = true; - LogEntryEventBuilder.addMockDataMaskRule(rule); - + rule.IsEnabled__c = true; + LogEntryEventBuilder.setMockDataMaskRule(rule); String message = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; - String expectedSanitizedMessage = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; Account account = new Account(Name = message); String accountJson = JSON.serializePretty(account); - String expectedSanitizedAccountJson = JSON.serializePretty(new Account(Name = expectedSanitizedMessage)); + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setMessage(message).setRecord(account); System.assertEquals(false, builder.getLogEntryEvent().MessageMasked__c); System.assertEquals(message, builder.getLogEntryEvent().Message__c); - System.assertNotEquals(expectedSanitizedMessage, builder.getLogEntryEvent().Message__c); + System.assertEquals(false, builder.getLogEntryEvent().RecordJsonMasked__c); + System.assertEquals(accountJson, builder.getLogEntryEvent().RecordJson__c); + } + + @IsTest + static void it_should_apply_data_mask_rule_to_message_when_enabled() { + Logger.getUserSettings().IsDataMaskingEnabled__c = true; + LogEntryDataMaskRule__mdt rule = getSocialSecurityNumberDataMaskRule(); + rule.IsEnabled__c = true; + LogEntryEventBuilder.setMockDataMaskRule(rule); + String sensitiveString = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; + String expectedSanitizedString = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; + + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); + builder.setMessage(sensitiveString); + + System.assertEquals(true, builder.getLogEntryEvent().MessageMasked__c); + System.assertNotEquals(sensitiveString, builder.getLogEntryEvent().Message__c); + System.assertEquals(expectedSanitizedString, builder.getLogEntryEvent().Message__c); + } + + @IsTest + static void it_should_apply_data_mask_rule_to_record_json_when_enabled() { + Logger.getUserSettings().IsDataMaskingEnabled__c = true; + LogEntryDataMaskRule__mdt rule = getSocialSecurityNumberDataMaskRule(); + rule.IsEnabled__c = true; + LogEntryEventBuilder.setMockDataMaskRule(rule); + String sensitiveString = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; + String expectedSanitizedString = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; + Account account = new Account(Name = sensitiveString); + String accountJson = JSON.serializePretty(account); + String expectedSanitizedAccountJson = JSON.serializePretty(new Account(Name = expectedSanitizedString)); + + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); + builder.setRecord(account); + System.assertEquals(true, builder.getLogEntryEvent().RecordJsonMasked__c); System.assertNotEquals(accountJson, builder.getLogEntryEvent().RecordJson__c); System.assertEquals(expectedSanitizedAccountJson, builder.getLogEntryEvent().RecordJson__c); @@ -187,64 +196,58 @@ private class LogEntryEventBuilder_Tests { static void it_should_apply_data_mask_rule_to_record_list_json_when_enabled() { Logger.getUserSettings().IsDataMaskingEnabled__c = true; LogEntryDataMaskRule__mdt rule = getSocialSecurityNumberDataMaskRule(); - rule.ApplyToMessage__c = false; - rule.ApplyToRecordJson__c = true; - LogEntryEventBuilder.addMockDataMaskRule(rule); + rule.IsEnabled__c = true; + LogEntryEventBuilder.setMockDataMaskRule(rule); - String message = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; - String expectedSanitizedMessage = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; - List accounts = new List{ new Account(Name = message) }; + String sensitiveString = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; + String expectedSanitizedString = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; + List accounts = new List{ new Account(Name = sensitiveString) }; String accountListJson = JSON.serializePretty(accounts); - String expectedSanitizedAccountListJson = JSON.serializePretty(new List{ new Account(Name = expectedSanitizedMessage) }); + String expectedSanitizedAccountListJson = JSON.serializePretty(new List{ new Account(Name = expectedSanitizedString) }); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setMessage(message).setRecord(accounts); + builder.setRecord(accounts); - System.assertEquals(false, builder.getLogEntryEvent().MessageMasked__c); - System.assertEquals(message, builder.getLogEntryEvent().Message__c); - System.assertNotEquals(expectedSanitizedMessage, builder.getLogEntryEvent().Message__c); System.assertEquals(true, builder.getLogEntryEvent().RecordJsonMasked__c); System.assertNotEquals(accountListJson, builder.getLogEntryEvent().RecordJson__c); System.assertEquals(expectedSanitizedAccountListJson, builder.getLogEntryEvent().RecordJson__c); } @IsTest - static void it_should_not_data_mask_rule_when_disabled_for_message_and_json() { + static void it_should_apply_data_mask_rule_to_http_request_body_when_enabled() { Logger.getUserSettings().IsDataMaskingEnabled__c = true; LogEntryDataMaskRule__mdt rule = getSocialSecurityNumberDataMaskRule(); - rule.ApplyToMessage__c = false; - rule.ApplyToRecordJson__c = false; - LogEntryEventBuilder.addMockDataMaskRule(rule); + rule.IsEnabled__c = true; + LogEntryEventBuilder.setMockDataMaskRule(rule); + String sensitiveString = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; + String expectedSanitizedString = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; + HttpRequest httpRequest = LoggerMockDataCreator.createHttpRequest(); + httpRequest.setBody(sensitiveString); - String message = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; - Account account = new Account(Name = message); - String accountJson = JSON.serializePretty(account); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setMessage(message).setRecord(account); + builder.setHttpRequestDetails(httpRequest); - System.assertEquals(false, builder.getLogEntryEvent().MessageMasked__c); - System.assertEquals(message, builder.getLogEntryEvent().Message__c); - System.assertEquals(false, builder.getLogEntryEvent().RecordJsonMasked__c); - System.assertEquals(accountJson, builder.getLogEntryEvent().RecordJson__c); + System.assertEquals(true, builder.getLogEntryEvent().HttpRequestBodyMasked__c); + System.assertNotEquals(sensitiveString, builder.getLogEntryEvent().HttpRequestBody__c); + System.assertEquals(expectedSanitizedString, builder.getLogEntryEvent().HttpRequestBody__c); } @IsTest - static void it_should_not_data_mask_rule_when_disabled_for_user() { - Logger.getUserSettings().IsDataMaskingEnabled__c = false; + static void it_should_apply_data_mask_rule_to_http_response_body_when_enabled() { + Logger.getUserSettings().IsDataMaskingEnabled__c = true; LogEntryDataMaskRule__mdt rule = getSocialSecurityNumberDataMaskRule(); - rule.ApplyToMessage__c = true; - rule.ApplyToRecordJson__c = true; - LogEntryEventBuilder.addMockDataMaskRule(rule); + rule.IsEnabled__c = true; + LogEntryEventBuilder.setMockDataMaskRule(rule); + String sensitiveString = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; + String expectedSanitizedString = 'Something, something, and my social is XXX-XX-9999 in case you want to steal my identity'; + HttpResponse httpResponse = LoggerMockDataCreator.createHttpResponse(); + httpResponse.setBody(sensitiveString); - String message = 'Something, something, and my social is 400 11 9999 in case you want to steal my identity'; - Account account = new Account(Name = message); - String accountJson = JSON.serializePretty(account); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setMessage(message).setRecord(account); + builder.setHttpResponseDetails(httpResponse); - System.assertEquals(false, builder.getLogEntryEvent().MessageMasked__c); - System.assertEquals(message, builder.getLogEntryEvent().Message__c); - System.assertEquals(false, builder.getLogEntryEvent().RecordJsonMasked__c); - System.assertEquals(accountJson, builder.getLogEntryEvent().RecordJson__c); + System.assertEquals(true, builder.getLogEntryEvent().HttpResponseBodyMasked__c); + System.assertNotEquals(sensitiveString, builder.getLogEntryEvent().HttpResponseBody__c); + System.assertEquals(expectedSanitizedString, builder.getLogEntryEvent().HttpResponseBody__c); } @IsTest @@ -295,379 +298,166 @@ private class LogEntryEventBuilder_Tests { } @IsTest - static void it_should_set_database_result_fields_for_successful_deleteResult() { - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - - Database.DeleteResult deleteResult = Database.delete(log, false); - System.assertEquals(true, deleteResult.isSuccess()); + static void it_should_set_database_result_fields_for_deleteResult() { + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); + System.assertNotEquals(null, deleteResult); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(deleteResult); + System.assertEquals(1, builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(deleteResult), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.DeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_failed_deleteResult() { - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - // Delete the log twice to trigger a DeleteResult error - delete log; - Database.DeleteResult deleteResult = Database.delete(log, false); - System.assertEquals(false, deleteResult.isSuccess()); + static void it_should_set_database_result_fields_for_mergeResult() { + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); + System.assertNotEquals(null, mergeResult); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(deleteResult); + builder.setDatabaseResult(mergeResult); + System.assertEquals(1, builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(deleteResult), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.DeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); + System.assertEquals(JSON.serializePretty(mergeResult), builder.getLogEntryEvent().DatabaseResultJson__c); + System.assertEquals(Database.MergeResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } - // TODO only some standard objects can be merged - need to find a way to either mock or trigger a merge result that works in all orgs - // @IsTest - // static void it_should_set_database_result_fields_for_failed_mergeResult() { - // Database.MergeResult mergeResult = Database.merge(new Account(Name='Expcted'), new Account(Name='Failure'), false); - // System.assertEquals(false, mergeResult.isSuccess()); - - // LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - - // Test.startTest(); - // builder.setDatabaseResult(mergeResult); - // Test.stopTest(); - - // LogEntryEvent__e logEntryEvent = builder.getLogEntryEvent(); - - // System.assertEquals(JSON.serializePretty(mergeResult), logEntryEvent.DatabaseResultJson__c); - // System.assertEquals(Database.MergeResult.class.getName(), logEntryEvent.DatabaseResultType__c); - // } - @IsTest - static void it_should_set_database_result_fields_for_successful_saveResult() { - Database.SaveResult saveResult = Database.insert(new Log__c(TransactionId__c = '1234'), false); - System.assertEquals(true, saveResult.isSuccess()); + static void it_should_set_database_result_fields_for_saveResult() { + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); + System.assertNotEquals(null, saveResult); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(saveResult); + System.assertEquals(1, builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(saveResult), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.SaveResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_failed_saveResult() { - Database.SaveResult saveResult = Database.insert(new Account(), false); - System.assertEquals(false, saveResult.isSuccess()); - - LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(saveResult); - - System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(saveResult), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.SaveResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); - } - - @IsTest - static void it_should_set_database_result_fields_for_successful_undeleteResult() { - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - delete log; - - Database.UndeleteResult undeleteResult = Database.undelete(log.Id, false); - System.assertEquals(true, undeleteResult.isSuccess()); + static void it_should_set_database_result_fields_for_undeleteResult() { + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); + System.assertNotEquals(null, undeleteResult); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(undeleteResult); + System.assertEquals(1, builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(undeleteResult), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.UndeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_failed_undeleteResult() { - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - delete log; - // Hard delete to (later) cause an undelete error - Database.emptyRecycleBin(log); - - Database.UndeleteResult undeleteResult = Database.undelete(log.Id, false); - System.assertEquals(false, undeleteResult.isSuccess()); - - LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(undeleteResult); - - System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(undeleteResult), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.UndeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); - } - - @IsTest - static void it_should_set_database_result_fields_for_successful_upsertResult_when_insert() { - // UpsertResult for a new record insert - Database.UpsertResult upsertResult = Database.upsert(new Log__c(TransactionId__c = '1234'), false); - System.assertEquals(true, upsertResult.isSuccess()); - - LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(upsertResult); - - System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(upsertResult), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.UpsertResult.class.getName() + '.Insert', builder.getLogEntryEvent().DatabaseResultType__c); - } - - @IsTest - static void it_should_set_database_result_fields_for_failed_upsertResult_when_insert() { - // UpsertResult for a new record insert - Database.UpsertResult upsertResult = Database.upsert(new Account(), false); - System.assertEquals(false, upsertResult.isSuccess()); + static void it_should_set_database_result_fields_for_upsertResult_when_insert() { + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); + System.assertNotEquals(null, upsertResult); + System.assertEquals(true, upsertResult.isCreated()); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(upsertResult); + System.assertEquals(1, builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(upsertResult), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.UpsertResult.class.getName() + '.Insert', builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_successful_upsertResult_when_update() { - // UpsertResult for an existing record update - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - log.TransactionId__c = '9876'; - - Database.UpsertResult upsertResult = Database.upsert(log, false); - System.assertEquals(true, upsertResult.isSuccess()); + static void it_should_set_database_result_fields_for_upsertResult_when_update() { + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); + System.assertNotEquals(null, upsertResult); + System.assertEquals(false, upsertResult.isCreated()); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(upsertResult); + System.assertEquals(1, builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(upsertResult), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.UpsertResult.class.getName() + '.Update', builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_failed_upsertResult_when_update() { - // UpsertResult for an existing record update - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - - // Use a log entry this time to handle the upsert scenario - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, Name = 'Test Log Entry'); - insert logEntry; - - // Set name field to a value too long for the field - Integer maxFieldLength = Schema.LogEntry__c.Name.getDescribe().getLength(); - logEntry.Name = '0'.repeat(maxFieldLength + 10); - - Database.UpsertResult upsertResult = Database.upsert(logEntry, false); - System.assertEquals(false, upsertResult.isSuccess()); - - LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(upsertResult); - - System.assertEquals('Single', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(upsertResult), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.UpsertResult.class.getName() + '.Update', builder.getLogEntryEvent().DatabaseResultType__c); - } - - @IsTest - static void it_should_set_database_result_fields_for_list_of_successful_deleteResult() { - List logs = new List(); + static void it_should_set_database_result_fields_for_list_of_deleteResult() { + List deleteResults = new List(); for (Integer i = 0; i < 5; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - insert logs; - - List deleteResults = Database.delete(logs, false); - for (Database.DeleteResult deleteResult : deleteResults) { - System.assertEquals(true, deleteResult.isSuccess()); + deleteResults.add(LoggerMockDataCreator.createDatabaseDeleteResult(false)); } LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(deleteResults); + System.assertEquals(deleteResults.size(), builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(deleteResults), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.DeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_list_of_failed_deleteResult() { - List logs = new List(); + static void it_should_set_database_result_fields_for_list_of_mergeResult() { + List mergeResults = new List(); for (Integer i = 0; i < 5; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - insert logs; - - // Delete the logs twice to trigger DeleteResult errors - delete logs; - - List deleteResults = Database.delete(logs, false); - for (Database.DeleteResult deleteResult : deleteResults) { - System.assertEquals(false, deleteResult.isSuccess()); + mergeResults.add(LoggerMockDataCreator.createDatabaseMergeResult(false)); } LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(deleteResults); + builder.setDatabaseResult(mergeResults); + System.assertEquals(mergeResults.size(), builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(deleteResults), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.DeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); + System.assertEquals(JSON.serializePretty(mergeResults), builder.getLogEntryEvent().DatabaseResultJson__c); + System.assertEquals(Database.MergeResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } - // TODO only some standard objects can be merged - need to find a way to either mock or trigger a merge result that works in all orgs - // @IsTest - // static void it_should_set_database_result_fields_for_list_of_failed_mergeResult() { - // Database.MergeResult mergeResult = Database.merge(new Account(Name='Expcted'), new Account(Name='Failure'), false); - // System.assertEquals(false, mergeResult.isSuccess()); - - // LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - - // Test.startTest(); - // builder.setDatabaseResult(mergeResult); - // Test.stopTest(); - - // LogEntryEvent__e logEntryEvent = builder.getLogEntryEvent(); - - // System.assertEquals(JSON.serializePretty(mergeResult), logEntryEvent.DatabaseResultJson__c); - // System.assertEquals(Database.MergeResult.class.getName(), logEntryEvent.DatabaseResultType__c); - // } - @IsTest - static void it_should_set_database_result_fields_for_list_of_successful_saveResult() { - List logs = new List(); + static void it_should_set_database_result_fields_for_list_of_saveResult() { + List saveResults = new List(); for (Integer i = 0; i < 5; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - List saveResults = Database.insert(logs, false); - for (Database.SaveResult saveResult : saveResults) { - System.assertEquals(true, saveResult.isSuccess()); + saveResults.add(LoggerMockDataCreator.createDatabaseSaveResult(false)); } LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(saveResults); + System.assertEquals(saveResults.size(), builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(saveResults), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.SaveResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_list_of_failed_saveResult() { - List accounts = new List(); + static void it_should_set_database_result_fields_for_list_of_undeleteResult() { + List undeleteResults = new List(); for (Integer i = 0; i < 5; i++) { - Account account = new Account(); - accounts.add(account); - } - List saveResults = Database.insert(accounts, false); - for (Database.SaveResult saveResult : saveResults) { - System.assertEquals(false, saveResult.isSuccess()); - } - - LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(saveResults); - - System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(saveResults), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.SaveResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); - } - - @IsTest - static void it_should_set_database_result_fields_for_list_of_successful_undeleteResult() { - List logs = new List(); - for (Integer i = 0; i < 5; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - insert logs; - delete logs; - - List undeleteResults = Database.undelete(logs, false); - for (Database.UndeleteResult undeleteResult : undeleteResults) { - System.assertEquals(true, undeleteResult.isSuccess()); + undeleteResults.add(LoggerMockDataCreator.createDatabaseUndeleteResult(false)); } LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(undeleteResults); + System.assertEquals(undeleteResults.size(), builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(undeleteResults), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.UndeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); } @IsTest - static void it_should_set_database_result_fields_for_list_of_failed_undeleteResult() { - List logs = new List(); + static void it_should_set_database_result_fields_for_list_of_upsertResult() { + List upsertResults = new List(); for (Integer i = 0; i < 5; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - insert logs; - delete logs; - // Hard delete to (later) cause an undelete error - Database.emptyRecycleBin(logs); - - List undeleteResults = Database.undelete(logs, false); - for (Database.UndeleteResult undeleteResult : undeleteResults) { - System.assertEquals(false, undeleteResult.isSuccess()); - } - - LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(undeleteResults); - - System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(undeleteResults), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.UndeleteResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); - } - - @IsTest - static void it_should_set_database_result_fields_for_list_of_successful_upsertResult_when_insert() { - // UpsertResult for a new record insert - List logs = new List(); - for (Integer i = 0; i < 5; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - List upsertResults = Database.upsert(logs, false); - for (Database.UpsertResult upsertResult : upsertResults) { - System.assertEquals(true, upsertResult.isSuccess()); - } - - LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); - builder.setDatabaseResult(upsertResults); - - System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); - System.assertEquals(JSON.serializePretty(upsertResults), builder.getLogEntryEvent().DatabaseResultJson__c); - System.assertEquals(Database.UpsertResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); - } - - @IsTest - static void it_should_set_database_result_fields_for_list_of_failed_upsertResult_when_insert() { - // UpsertResult for a new record insert - List accounts = new List(); - for (Integer i = 0; i < 5; i++) { - Account account = new Account(); - accounts.add(account); - } - List upsertResults = Database.upsert(accounts, false); - for (Database.UpsertResult upsertResult : upsertResults) { - System.assertEquals(false, upsertResult.isSuccess()); + upsertResults.add(LoggerMockDataCreator.createDatabaseUpsertResult(false, false)); } LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); builder.setDatabaseResult(upsertResults); + System.assertEquals(upsertResults.size(), builder.getLogEntryEvent().DatabaseResultCollectionSize__c); System.assertEquals('List', builder.getLogEntryEvent().DatabaseResultCollectionType__c); System.assertEquals(JSON.serializePretty(upsertResults), builder.getLogEntryEvent().DatabaseResultJson__c); System.assertEquals(Database.UpsertResult.class.getName(), builder.getLogEntryEvent().DatabaseResultType__c); @@ -685,6 +475,7 @@ private class LogEntryEventBuilder_Tests { Id currentUserId = UserInfo.getUserId(); builder.setRecordId(currentUserId); + System.assertEquals(1, builder.getLogEntryEvent().RecordCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().RecordCollectionType__c); System.assertEquals(currentUserId, builder.getLogEntryEvent().RecordId__c); System.assertEquals(null, builder.getLogEntryEvent().RecordJson__c); @@ -702,12 +493,12 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectType__c); System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectTypeNamespace__c); - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - builder.setRecordId(log.Id); + Id mockLogId = LoggerMockDataCreator.createId(Schema.Log__c.SObjectType); + builder.setRecordId(mockLogId); + System.assertEquals(1, builder.getLogEntryEvent().RecordCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().RecordCollectionType__c); - System.assertEquals(log.Id, builder.getLogEntryEvent().RecordId__c); + System.assertEquals(mockLogId, builder.getLogEntryEvent().RecordId__c); System.assertEquals(null, builder.getLogEntryEvent().RecordJson__c); System.assertEquals('Custom Object', builder.getLogEntryEvent().RecordSObjectClassification__c); System.assertEquals(Log__c.SObjectType.getDescribe().getName(), builder.getLogEntryEvent().RecordSObjectType__c); @@ -722,13 +513,10 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectType__c); System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectTypeNamespace__c); - Test.startTest(); - LogStatus__mdt status = [SELECT Id, MasterLabel, DeveloperName FROM LogStatus__mdt LIMIT 1]; builder.setRecordId(status.Id); - Test.stopTest(); - + System.assertEquals(1, builder.getLogEntryEvent().RecordCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().RecordCollectionType__c); System.assertEquals(status.Id, builder.getLogEntryEvent().RecordId__c); System.assertEquals(null, builder.getLogEntryEvent().RecordJson__c); @@ -745,14 +533,11 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectType__c); System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectTypeNamespace__c); - Test.startTest(); - User nullUser; System.assertEquals(null, nullUser); builder.setRecordId(nullUser); - Test.stopTest(); - + System.assertEquals(1, builder.getLogEntryEvent().RecordCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().RecordCollectionType__c); System.assertEquals(null, builder.getLogEntryEvent().RecordId__c); System.assertEquals('null', builder.getLogEntryEvent().RecordJson__c); @@ -770,13 +555,10 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectType__c); System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectTypeNamespace__c); - Test.startTest(); - User currentUser = [SELECT Id, Name, ProfileId, Profile.Name, IsActive FROM User WHERE Id = :UserInfo.getUserId()]; builder.setRecordId(currentUser); - Test.stopTest(); - + System.assertEquals(1, builder.getLogEntryEvent().RecordCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().RecordCollectionType__c); System.assertEquals(currentUser.Id, builder.getLogEntryEvent().RecordId__c); System.assertEquals(JSON.serializePretty(currentUser), builder.getLogEntryEvent().RecordJson__c); @@ -794,14 +576,10 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectType__c); System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectTypeNamespace__c); - Test.startTest(); - - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); builder.setRecordId(log); - Test.stopTest(); - + System.assertEquals(1, builder.getLogEntryEvent().RecordCollectionSize__c); System.assertEquals('Single', builder.getLogEntryEvent().RecordCollectionType__c); System.assertEquals(log.Id, builder.getLogEntryEvent().RecordId__c); System.assertEquals(JSON.serializePretty(log), builder.getLogEntryEvent().RecordJson__c); @@ -818,13 +596,11 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectType__c); System.assertEquals(null, builder.getLogEntryEvent().RecordSObjectTypeNamespace__c); - Test.startTest(); - LogStatus__mdt status = [SELECT Id, MasterLabel, DeveloperName FROM LogStatus__mdt LIMIT 1]; builder.setRecordId(status); - Test.stopTest(); - + System.assertEquals(1, builder.getLogEntryEvent().RecordCollectionSize__c); + System.assertEquals('Single', builder.getLogEntryEvent().RecordCollectionType__c); System.assertEquals(status.Id, builder.getLogEntryEvent().RecordId__c); System.assertEquals(JSON.serializePretty(status), builder.getLogEntryEvent().RecordJson__c); System.assertEquals('Custom Metadata Type Object', builder.getLogEntryEvent().RecordSObjectClassification__c); @@ -833,8 +609,8 @@ private class LogEntryEventBuilder_Tests { @IsTest static void it_should_skip_stripping_inaccessible_fields_for_aggregate_result() { - User standardUser = LoggerTestUtils.createUser(); - AggregateResult mockAggregateResult = LoggerTestUtils.createMockAggregateResult(); + User standardUser = LoggerMockDataCreator.createUser(); + AggregateResult mockAggregateResult = LoggerMockDataCreator.createAggregateResult(); LogEntryEventBuilder builder; System.runAs(standardUser) { @@ -847,8 +623,8 @@ private class LogEntryEventBuilder_Tests { @IsTest static void it_should_skip_stripping_inaccessible_fields_for_aggregate_results() { - User standardUser = LoggerTestUtils.createUser(); - List mockAggregateResults = new List{ LoggerTestUtils.createMockAggregateResult() }; + User standardUser = LoggerMockDataCreator.createUser(); + List mockAggregateResults = new List{ LoggerMockDataCreator.createAggregateResult() }; LogEntryEventBuilder builder; System.runAs(standardUser) { @@ -940,6 +716,80 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(User.SObjectType.getDescribe().getName(), builder.getLogEntryEvent().RecordSObjectType__c); } + @IsTest + static void it_should_skip_setting_http_request_fields_when_request_is_null() { + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestBody__c); + System.assertEquals(false, builder.getLogEntryEvent().HttpRequestCompressed__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestEndpoint__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestMethod__c); + HttpRequest request = null; + + builder.setHttpRequestDetails(request); + + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestBody__c); + System.assertEquals(false, builder.getLogEntryEvent().HttpRequestCompressed__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestEndpoint__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestMethod__c); + } + + @IsTest + static void it_should_set_http_request_fields_when_request_is_populated() { + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestBody__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestEndpoint__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestMethod__c); + HttpRequest request = new HttpRequest(); + request.setBody('Hello, world!'); + request.setCompressed(true); + request.setEndpoint('https://fake.salesforce.com'); + request.setMethod('GET'); + + builder.setHttpRequestDetails(request); + + System.assertEquals(request.getBody(), builder.getLogEntryEvent().HttpRequestBody__c); + System.assertEquals(request.getCompressed(), builder.getLogEntryEvent().HttpRequestCompressed__c); + System.assertEquals(request.getEndpoint(), builder.getLogEntryEvent().HttpRequestEndpoint__c); + System.assertEquals(request.getMethod(), builder.getLogEntryEvent().HttpRequestMethod__c); + } + + @IsTest + static void it_should_skip_setting_http_response_fields_when_response_is_null() { + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestBody__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestEndpoint__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestMethod__c); + HttpResponse response = null; + + builder.setHttpResponseDetails(response); + + System.assertEquals(null, builder.getLogEntryEvent().HttpResponseBody__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpResponseHeaderKeys__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpResponseStatus__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpResponseStatusCode__c); + } + + @IsTest + static void it_should_set_http_response_fields_when_populated() { + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestBody__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestEndpoint__c); + System.assertEquals(null, builder.getLogEntryEvent().HttpRequestMethod__c); + HttpResponse response = new HttpResponse(); + response.setBody('Hello, world!'); + response.setHeader('someKey', 'some string value'); + response.setHeader('anotherKey', 'an amazing example value, wow'); + response.setStatus('STATUS_GOOD_JOB_YOU_DID_IT'); + response.setStatusCode(201); + + builder.setHttpResponseDetails(response); + + System.assertEquals(response.getBody(), builder.getLogEntryEvent().HttpResponseBody__c); + System.assertEquals(String.join(response.getHeaderKeys(), '\n'), builder.getLogEntryEvent().HttpResponseHeaderKeys__c); + System.assertEquals(response.getStatus(), builder.getLogEntryEvent().HttpResponseStatus__c); + System.assertEquals(response.getStatusCode(), builder.getLogEntryEvent().HttpResponseStatusCode__c); + } + @IsTest static void it_should_set_tags_string_for_list_of_tags() { LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); @@ -948,19 +798,20 @@ private class LogEntryEventBuilder_Tests { List tags = new List{ 'some-tag', 'another One', 'here\'s one more!' }; builder.addTags(tags); + tags.sort(); String expectedTagsString = String.escapeSingleQuotes(String.join(tags, '\n')); System.assertEquals(expectedTagsString, builder.getLogEntryEvent().Tags__c); } @IsTest - static void it_should_deduplicate_tags() { + static void it_should_deduplicate_and_sort_tags() { LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); System.assertEquals(null, builder.getLogEntryEvent().Tags__c); List tags = new List{ 'duplicate-tag', 'duplicate-tag', 'another One' }; builder.addTags(tags); - String expectedTagsString = 'duplicate-tag\nanother One'; + String expectedTagsString = 'another One\nduplicate-tag'; System.assertEquals(expectedTagsString, builder.getLogEntryEvent().Tags__c); } @@ -998,6 +849,23 @@ private class LogEntryEventBuilder_Tests { System.assertEquals(anonymousApexStackTraceString, builder.getLogEntryEvent().StackTrace__c); } + @IsTest + static void it_should_deduplicate_sequential_lines_when_setting_stack_trace() { + String anonymousApexOriginLocation = 'AnonymousBlock'; + String deduplicatedStackTraceString = 'AnonymousBlock: line 9, column 1'; + String duplicatedStackTraceString = deduplicatedStackTraceString + '\n' + deduplicatedStackTraceString + '\n' + deduplicatedStackTraceString; + + LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.FINE, true); + builder.getLogEntryEvent().OriginLocation__c = null; + builder.getLogEntryEvent().StackTrace__c = null; + builder.parseStackTrace(duplicatedStackTraceString); + + System.assertNotEquals(null, builder.getLogEntryEvent().OriginLocation__c); + System.assertEquals(anonymousApexOriginLocation, builder.getLogEntryEvent().OriginLocation__c); + System.assertNotEquals(null, builder.getLogEntryEvent().StackTrace__c); + System.assertEquals(deduplicatedStackTraceString, builder.getLogEntryEvent().StackTrace__c); + } + @IsTest static void it_should_not_set_stack_trace_and_origin_location_for_invalid_stack_trace_string() { final String invalidStackTrace = '()'; @@ -1027,9 +895,9 @@ private class LogEntryEventBuilder_Tests { @IsTest static void it_should_set_detailed_fields() { // Get expected data - Organization organization = LoggerTestUtils.getOrganization(); - String organizationEnvironmentType = LoggerTestUtils.getOrganizationEnvironmentType(); - User user = LoggerTestUtils.getCurrentUser(); + Organization organization = LoggerMockDataCreator.getOrganization(); + String organizationEnvironmentType = LoggerMockDataCreator.getOrganizationEnvironmentType(); + User user = LoggerMockDataCreator.getUser(); LogEntryEventBuilder builder = new LogEntryEventBuilder(LoggingLevel.INFO, true); @@ -1054,7 +922,7 @@ private class LogEntryEventBuilder_Tests { @IsTest static void it_should_use_configured_log_entry_event_fields_for_debug_string() { - LoggerParameter.setMockParameter( + LoggerParameter.setMock( new LoggerParameter__mdt(DeveloperName = 'SystemDebugMessageFormat', Value__c = '{OriginLocation__c}\n{Message__c}: {LoggingLevel__c}') ); @@ -1072,6 +940,23 @@ private class LogEntryEventBuilder_Tests { ); } + static String getMessage() { + return 'Hello, world'; + } + + static LogMessage getLogMessage() { + return new LogMessage('The current date is {0}', System.today()); + } + + static LogEntryDataMaskRule__mdt getSocialSecurityNumberDataMaskRule() { + return new LogEntryDataMaskRule__mdt( + DeveloperName = 'SocialSecurityNumber', + IsEnabled__c = true, + SensitiveDataRegEx__c = '(\\d{3})[- ]*(\\d{2})[- ]*(\\d{4})', + ReplacementRegEx__c = 'XXX-XX-$3' + ); + } + private class DebugStringExample { public final String loggingString = 'Inside DebugStringExample.myMethod !'; public LogEntryEventBuilder myMethod() { diff --git a/nebula-logger/core/tests/logger-engine/classes/LogMessage_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/LogMessage_Tests.cls index a7454f265..bf77ee293 100644 --- a/nebula-logger/core/tests/logger-engine/classes/LogMessage_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/LogMessage_Tests.cls @@ -4,12 +4,13 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LogMessage_Tests { - public class CustomLogMessage extends LogMessage { + private static final String CUSTOM_MESSAGE_PREFIX = 'Just confirming that CustomLogMessage class is used'; + private class CustomLogMessage extends LogMessage { private String inputString; public CustomLogMessage(String inputString) { - this.inputString = inputString; + this.inputString = CUSTOM_MESSAGE_PREFIX + inputString; } public override String getMessage() { return inputString; @@ -18,14 +19,10 @@ private class LogMessage_Tests { @IsTest static void it_should_support_extending_LogMessage_class() { - String expectedMessage = 'The current time is: ' + System.now(); + String originalMessage = CUSTOM_MESSAGE_PREFIX + 'The current time is: ' + System.now(); + String expectedMessage = CUSTOM_MESSAGE_PREFIX + originalMessage; - Test.startTest(); - - CustomLogMessage customLogMessage = new CustomLogMessage(expectedMessage); - String returnedMessage = customLogMessage.getMessage(); - - Test.stopTest(); + String returnedMessage = new CustomLogMessage(originalMessage).getMessage(); System.assertEquals(expectedMessage, returnedMessage); } @@ -36,9 +33,7 @@ private class LogMessage_Tests { List arguments = new List{ System.now() }; String expectedMessage = String.format(unformattedMessage, arguments); - Test.startTest(); String returnedMessage = new LogMessage(unformattedMessage, arguments).getMessage(); - Test.stopTest(); System.assertEquals(expectedMessage, returnedMessage); } @@ -49,9 +44,7 @@ private class LogMessage_Tests { Datetime argument1 = System.now(); String expectedMessage = String.format(unformattedMessage, new List{ argument1 }); - Test.startTest(); String returnedMessage = new LogMessage(unformattedMessage, argument1).getMessage(); - Test.stopTest(); System.assertEquals(expectedMessage, returnedMessage); } @@ -63,9 +56,7 @@ private class LogMessage_Tests { Date argument2 = System.today(); String expectedMessage = String.format(unformattedMessage, new List{ argument1, argument2 }); - Test.startTest(); String returnedMessage = new LogMessage(unformattedMessage, argument1, argument2).getMessage(); - Test.stopTest(); System.assertEquals(expectedMessage, returnedMessage); } @@ -78,9 +69,7 @@ private class LogMessage_Tests { User argument3 = new User(Id = UserInfo.getUserId()); String expectedMessage = String.format(unformattedMessage, new List{ argument1, argument2, argument3 }); - Test.startTest(); String returnedMessage = new LogMessage(unformattedMessage, argument1, argument2, argument3).getMessage(); - Test.stopTest(); System.assertEquals(expectedMessage, returnedMessage); } diff --git a/nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls new file mode 100644 index 000000000..be2a9f514 --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls @@ -0,0 +1,608 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +// TODO switch to using standard object(s) available in all orgs (instead of Log__c) so this can be split off as a separate repo +// Possible objects to use: User, Group (Queue) and QueueSObject (none support undelete though), and not sure if it's possible +// to test platform events/EventBus using only standard objects +@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LoggerDataStore_Tests { + // Database DML tests + @IsTest + static void it_should_delete_record() { + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + insert log; + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + Database.DeleteResult result = LoggerDataStore.getDatabase().deleteRecord(log); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + System.assertEquals(true, result.isSuccess()); + Log__c persistedLog = [SELECT Id, IsDeleted FROM Log__c WHERE Id = :log.Id ALL ROWS]; + System.assertEquals(true, persistedLog.IsDeleted); + } + + @IsTest + static void it_should_delete_record_list() { + List logsToDelete = new List(); + logsToDelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToDelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + insert logsToDelete; + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().deleteRecords(logsToDelete); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(4, Limits.getDmlRows()); + System.assertEquals(logsToDelete.size(), results.size()); + for (Database.DeleteResult result : results) { + System.assertEquals(true, result.isSuccess()); + } + List persistedLogs = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToDelete ALL ROWS]; + System.assertEquals(logsToDelete.size(), persistedLogs.size()); + for (Log__c persistedLog : persistedLogs) { + System.assertEquals(true, persistedLog.IsDeleted); + } + } + + @IsTest + static void it_should_partially_delete_record_list_when_allOrNone_is_false() { + List logsToDelete = new List(); + logsToDelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToDelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + insert logsToDelete; + // Delete one of the logs first to cause a failure in the delete DML statement + delete logsToDelete.get(0); + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().deleteRecords(logsToDelete, false); + + System.assertEquals(3, Limits.getDmlStatements()); + System.assertEquals(5, Limits.getDmlRows()); + System.assertEquals(logsToDelete.size(), results.size()); + System.assertEquals(false, results.get(0).isSuccess()); + System.assertEquals(true, results.get(1).isSuccess()); + List persistedLogs = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToDelete ALL ROWS]; + System.assertEquals(logsToDelete.size(), persistedLogs.size()); + for (Log__c persistedLog : persistedLogs) { + System.assertEquals(true, persistedLog.IsDeleted); + } + } + + @IsTest + static void it_should_hard_delete_record() { + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + log.TransactionId__c = '999'; + insert log; + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + Database.DeleteResult result = LoggerDataStore.getDatabase().hardDeleteRecord(log); + + System.assertEquals(3, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + System.assertEquals(true, result.isSuccess()); + // Frustratingly, there's not a way to ensure that Database.emptyRecycleBin() was called and + // the supposedly-hard-deleted records are still returned in SOQL, so at least verify that the + // record has IsDeleted = true for now. This can probably be tested better with a @TestVisible property later. + Log__c persistedLog = [SELECT Id, IsDeleted FROM Log__c WHERE Id = :log.Id ALL ROWS]; + System.assertEquals(true, persistedLog.IsDeleted); + } + + @IsTest + static void it_should_hard_delete_record_list() { + List logsToDelete = new List(); + logsToDelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToDelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + insert logsToDelete; + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().hardDeleteRecords(logsToDelete); + + System.assertEquals(3, Limits.getDmlStatements()); + System.assertEquals(6, Limits.getDmlRows()); + System.assertEquals(logsToDelete.size(), results.size()); + for (Database.DeleteResult result : results) { + System.assertEquals(true, result.isSuccess()); + } + // Frustratingly, there's not a way to ensure that Database.emptyRecycleBin() was called and + // the supposedly-hard-deleted records are still returned in SOQL, so at least verify that the + // record has IsDeleted = true for now. This can probably be tested better with a @TestVisible property later. + List persistedLogs = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToDelete ALL ROWS]; + System.assertEquals(logsToDelete.size(), persistedLogs.size()); + for (Log__c persistedLog : persistedLogs) { + System.assertEquals(true, persistedLog.IsDeleted); + } + } + + @IsTest + static void it_should_insert_record() { + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + log.TransactionId__c = '1234'; + System.assertEquals(0, Limits.getDmlStatements()); + System.assertEquals(0, Limits.getDmlRows()); + + Database.SaveResult result = LoggerDataStore.getDatabase().insertRecord(log); + + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + System.assertNotEquals(null, log.Id); + System.assertEquals(true, result.isSuccess()); + Log__c persistedLog = [SELECT Id, TransactionId__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(log.TransactionId__c, persistedLog.TransactionId__c); + } + + @IsTest + static void it_should_insert_record_list() { + List logsToInsert = new List(); + logsToInsert.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToInsert.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + System.assertEquals(0, Limits.getDmlStatements()); + System.assertEquals(0, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().insertRecords(logsToInsert); + + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + System.assertEquals(logsToInsert.size(), results.size()); + for (Database.SaveResult result : results) { + System.assertEquals(true, result.isSuccess()); + } + List persistedLogs = [SELECT Id, TransactionId__c FROM Log__c WHERE Id IN :logsToInsert]; + System.assertEquals(logsToInsert.size(), persistedLogs.size()); + } + + @IsTest + static void it_should_partially_insert_record_list_when_allOrNone_is_false() { + Log__c existingRecord = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + insert existingRecord; + List logsToInsert = new List(); + logsToInsert.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToInsert.add(existingRecord); + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().insertRecords(logsToInsert, false); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + System.assertEquals(logsToInsert.size(), results.size()); + System.assertEquals(true, results.get(0).isSuccess()); + System.assertEquals(false, results.get(1).isSuccess()); + List persistedLogs = [SELECT Id, TransactionId__c FROM Log__c WHERE Id IN :logsToInsert]; + System.assertEquals(2, persistedLogs.size()); + } + + @IsTest + static void it_should_partially_insert_record_list_when_allOrNone_is_false_in_dml_options() { + Log__c existingRecord = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + insert existingRecord; + List logsToInsert = new List(); + logsToInsert.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToInsert.add(existingRecord); + Database.DmlOptions dmlOptions = new Database.DmlOptions(); + dmlOptions.OptAllOrNone = false; + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().insertRecords(logsToInsert, dmlOptions); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + System.assertEquals(logsToInsert.size(), results.size()); + System.assertEquals(true, results.get(0).isSuccess()); + System.assertEquals(false, results.get(1).isSuccess()); + List persistedLogs = [SELECT Id, TransactionId__c FROM Log__c WHERE Id IN :logsToInsert]; + System.assertEquals(2, persistedLogs.size()); + } + + @IsTest + static void it_should_undelete_record() { + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + insert log; + delete log; + log = [SELECT Id, IsDeleted FROM Log__c WHERE Id = :log.Id ALL ROWS]; + System.assertEquals(true, log.IsDeleted); + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + + Database.UndeleteResult result = LoggerDataStore.getDatabase().undeleteRecord(log); + + System.assertEquals(3, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + System.assertEquals(true, result.isSuccess()); + Log__c persistedLog = [SELECT Id, IsDeleted FROM Log__c WHERE Id = :log.Id ALL ROWS]; + System.assertEquals(false, persistedLog.IsDeleted); + } + + @IsTest + static void it_should_undelete_records() { + List logsToUndelete = new List(); + logsToUndelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToUndelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + insert logsToUndelete; + delete logsToUndelete; + logsToUndelete = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToUndelete ALL ROWS]; + for (Log__c deletedLog : logsToUndelete) { + System.assertEquals(true, deletedLog.IsDeleted); + } + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(4, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().undeleteRecords(logsToUndelete); + + System.assertEquals(3, Limits.getDmlStatements()); + System.assertEquals(6, Limits.getDmlRows()); + System.assertEquals(logsToUndelete.size(), results.size()); + for (Database.UndeleteResult result : results) { + System.assertEquals(true, result.isSuccess()); + } + List persistedLogs = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToUndelete ALL ROWS]; + System.assertEquals(logsToUndelete.size(), persistedLogs.size()); + for (Log__c persistedLog : persistedLogs) { + System.assertEquals(false, persistedLog.IsDeleted); + } + } + + @IsTest + static void it_should_partially_undelete_records_when_allOrNone_is_false() { + List logsToUndelete = new List(); + logsToUndelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + logsToUndelete.add((Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord()); + insert logsToUndelete; + delete logsToUndelete.get(0); + logsToUndelete = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToUndelete ALL ROWS]; + System.assertEquals(true, logsToUndelete.get(0).IsDeleted); + System.assertEquals(false, logsToUndelete.get(1).IsDeleted); + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().undeleteRecords(logsToUndelete, false); + + System.assertEquals(3, Limits.getDmlStatements()); + System.assertEquals(5, Limits.getDmlRows()); + System.assertEquals(logsToUndelete.size(), results.size()); + System.assertEquals(true, results.get(0).isSuccess()); + System.assertEquals(false, results.get(1).isSuccess()); + List persistedLogs = [SELECT Id, IsDeleted FROM Log__c WHERE Id IN :logsToUndelete ALL ROWS]; + System.assertEquals(logsToUndelete.size(), persistedLogs.size()); + for (Log__c persistedLog : persistedLogs) { + System.assertEquals(false, persistedLog.IsDeleted); + } + } + + @IsTest + static void it_should_update_record() { + String originalScenario = 'Some scenario'; + String updatedScenario = 'Another, different scenario'; + List logsToUpdate = new List(); + logsToUpdate.add((Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario)).populateRequiredFields().getRecord()); + logsToUpdate.add((Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario)).populateRequiredFields().getRecord()); + insert logsToUpdate; + logsToUpdate = [SELECT Id, Scenario__c FROM Log__c WHERE Id IN :logsToUpdate]; + for (Log__c logToUpdate : logsToUpdate) { + System.assertEquals(originalScenario, logToUpdate.Scenario__c); + logToUpdate.Scenario__c = updatedScenario; + } + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().updateRecords(logsToUpdate); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(4, Limits.getDmlRows()); + System.assertEquals(logsToUpdate.size(), results.size()); + for (Database.SaveResult result : results) { + System.assertEquals(true, result.isSuccess()); + } + List persistedLogs = [SELECT Id, Scenario__c FROM Log__c WHERE Id IN :logsToUpdate ALL ROWS]; + System.assertEquals(logsToUpdate.size(), persistedLogs.size()); + for (Log__c persistedLog : persistedLogs) { + System.assertEquals(updatedScenario, persistedLog.Scenario__c); + } + } + + @IsTest + static void it_should_update_record_list() { + String originalScenario = 'Some scenario'; + String updatedScenario = 'Another, different scenario'; + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + log.Scenario__c = originalScenario; + insert log; + log = [SELECT Id, Scenario__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalScenario, log.Scenario__c); + log.Scenario__c = updatedScenario; + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + Database.SaveResult result = LoggerDataStore.getDatabase().updateRecord(log); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + System.assertEquals(true, result.isSuccess()); + Log__c persistedLog = [SELECT Id, Scenario__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(updatedScenario, persistedLog.Scenario__c); + } + + @IsTest + static void it_should_partially_update_record_list_when_allOrNone_is_false() { + String originalScenario = 'Some scenario'; + String updatedScenario = 'Another, different scenario'; + List logsToUpdate = new List(); + logsToUpdate.add((Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario)).populateRequiredFields().getRecord()); + insert logsToUpdate; + logsToUpdate.add((Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario)).populateRequiredFields().getRecord()); + for (Log__c log : logsToUpdate) { + log.Scenario__c = updatedScenario; + } + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().updateRecords(logsToUpdate, false); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + System.assertEquals(logsToUpdate.size(), results.size()); + System.assertEquals(true, results.get(0).isSuccess()); + System.assertEquals(false, results.get(1).isSuccess()); + List persistedLogs = [SELECT Id, Scenario__c FROM Log__c WHERE Id IN :logsToUpdate]; + System.assertEquals(1, persistedLogs.size()); + System.assertEquals(updatedScenario, persistedLogs.get(0).Scenario__c); + } + + @IsTest + static void it_should_partially_update_record_list_when_allOrNone_is_false_in_dml_options() { + String originalScenario = 'Some scenario'; + String updatedScenario = 'Another, different scenario'; + List logsToUpdate = new List(); + logsToUpdate.add((Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario)).populateRequiredFields().getRecord()); + insert logsToUpdate; + logsToUpdate.add((Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario)).populateRequiredFields().getRecord()); + for (Log__c log : logsToUpdate) { + log.Scenario__c = updatedScenario; + } + Database.DmlOptions dmlOptions = new Database.DmlOptions(); + dmlOptions.OptAllOrNone = false; + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().updateRecords(logsToUpdate, dmlOptions); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(3, Limits.getDmlRows()); + System.assertEquals(logsToUpdate.size(), results.size()); + System.assertEquals(true, results.get(0).isSuccess()); + System.assertEquals(false, results.get(1).isSuccess()); + List persistedLogs = [SELECT Id, Scenario__c FROM Log__c WHERE Id IN :logsToUpdate]; + System.assertEquals(1, persistedLogs.size()); + System.assertEquals(updatedScenario, persistedLogs.get(0).Scenario__c); + } + + @IsTest + static void it_should_upsert_record() { + String originalScenario = 'Some scenario'; + String updatedScenario = 'Another, different scenario'; + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + log.Scenario__c = originalScenario; + log.TransactionId__c = '1234'; + insert log; + log = [SELECT Id, Scenario__c, TransactionId__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalScenario, log.Scenario__c); + log.Scenario__c = updatedScenario; + Id originalLogId = log.Id; + log.Id = null; + System.assertEquals(null, log.Id); + System.assertNotEquals(null, log.TransactionId__c); + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(1, Limits.getDmlRows()); + + Database.UpsertResult result = LoggerDataStore.getDatabase().upsertRecord(log, Schema.Log__c.TransactionId__c); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + System.assertEquals(false, result.isCreated()); + System.assertEquals(true, result.isSuccess()); + Log__c persistedLog = [SELECT Id, Scenario__c, TransactionId__c FROM Log__c WHERE TransactionId__c = :log.TransactionId__c]; + System.assertEquals(originalLogId, persistedLog.Id); + System.assertEquals(updatedScenario, persistedLog.Scenario__c); + } + + @IsTest + static void it_should_upsert_record_list() { + String originalScenario = 'Some scenario'; + String updatedScenario = 'Another, different scenario'; + List logsToUpsert = new List(); + logsToUpsert.add( + (Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario, TransactionId__c = '1234')) + .populateRequiredFields() + .getRecord() + ); + logsToUpsert.add( + (Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario, TransactionId__c = '5678')) + .populateRequiredFields() + .getRecord() + ); + insert logsToUpsert; + logsToUpsert = [SELECT Id, Scenario__c, TransactionId__c FROM Log__c WHERE Id IN :logsToUpsert]; + for (Log__c logToUpdate : logsToUpsert) { + System.assertEquals(originalScenario, logToUpdate.Scenario__c); + logToUpdate.Scenario__c = updatedScenario; + } + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().upsertRecords(logsToUpsert, Schema.Log__c.TransactionId__c); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(4, Limits.getDmlRows()); + System.assertEquals(logsToUpsert.size(), results.size()); + for (Database.UpsertResult result : results) { + System.assertEquals(true, result.isSuccess()); + } + List persistedLogs = [SELECT Id, Scenario__c FROM Log__c WHERE Id IN :logsToUpsert]; + System.assertEquals(logsToUpsert.size(), persistedLogs.size()); + for (Log__c persistedLog : persistedLogs) { + System.assertEquals(updatedScenario, persistedLog.Scenario__c); + } + } + + @IsTest + static void it_should_partially_upsert_record_list_when_allOrNone_is_false() { + String originalScenario = 'Some scenario'; + String updatedScenario = 'Another, different scenario'; + List logsToUpsert = new List(); + logsToUpsert.add( + (Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario, TransactionId__c = '1234')) + .populateRequiredFields() + .getRecord() + ); + logsToUpsert.add( + (Log__c) LoggerMockDataCreator.createDataBuilder(new Log__c(Scenario__c = originalScenario, TransactionId__c = '5678')) + .populateRequiredFields() + .getRecord() + ); + insert logsToUpsert; + logsToUpsert = [SELECT Id, Scenario__c, TransactionId__c FROM Log__c WHERE Id IN :logsToUpsert]; + for (Log__c logToUpdate : logsToUpsert) { + System.assertEquals(originalScenario, logToUpdate.Scenario__c); + logToUpdate.Scenario__c = updatedScenario; + } + logsToUpsert.get(1).TransactionId__c = null; // This will cause an error since the upsert is on TransactionId__c + System.assertEquals(1, Limits.getDmlStatements()); + System.assertEquals(2, Limits.getDmlRows()); + + List results = LoggerDataStore.getDatabase().upsertRecords(logsToUpsert, Schema.Log__c.TransactionId__c, false); + + System.assertEquals(2, Limits.getDmlStatements()); + System.assertEquals(4, Limits.getDmlRows()); + System.assertEquals(logsToUpsert.size(), results.size()); + System.assertEquals(true, results.get(0).isSuccess()); + System.assertEquals(false, results.get(1).isSuccess()); + List persistedLogs = [SELECT Id, Scenario__c FROM Log__c WHERE Id IN :logsToUpsert]; + System.assertEquals(2, persistedLogs.size()); + System.assertEquals(updatedScenario, persistedLogs.get(0).Scenario__c); + System.assertEquals(originalScenario, persistedLogs.get(1).Scenario__c); + } + + // Event Bus tests + @IsTest + static void it_should_publish_record() { + LogEntryEvent__e logEntryEvent = (LOgEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType) + .populateRequiredFields() + .getRecord(); + System.assertEquals(0, Limits.getPublishImmediateDml()); + + Database.SaveResult result = LoggerDataStore.getEventBus().publishRecord(logEntryEvent); + + System.assertEquals(1, Limits.getPublishImmediateDml(), result); + System.assertEquals(true, result.isSuccess()); + } + + @IsTest + static void it_should_publish_record_list() { + List logEntryEventsToPublish = new List(); + logEntryEventsToPublish.add( + (LogEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType).populateRequiredFields().getRecord() + ); + logEntryEventsToPublish.add( + (LogEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType).populateRequiredFields().getRecord() + ); + System.assertEquals(0, Limits.getPublishImmediateDml()); + + List results = LoggerDataStore.getEventBus().publishRecords(logEntryEventsToPublish); + + System.assertEquals(1, Limits.getPublishImmediateDml(), results); + System.assertEquals(logEntryEventsToPublish.size(), results.size()); + for (Database.SaveResult result : results) { + System.assertEquals(true, result.isSuccess()); + } + } + + // Queueable tests + @IsTest + static void it_should_enqueue_job() { + System.assertEquals(0, Limits.getQueueableJobs()); + + LoggerDataStore.getJobQueue().enqueueJob(new MockQueueable()); + + System.assertEquals(1, Limits.getQueueableJobs()); + } + + // Mocking tests + @IsTest + static void it_should_support_overriding_database_instance() { + MockDatabase mockDatabaseInstance = new MockDatabase(); + System.assertEquals(0, mockDatabaseInstance.callCount); + LoggerDataStore.setMock(mockDatabaseInstance); + System.assertEquals(mockDatabaseInstance, LoggerDataStore.getDatabase()); + + LoggerDataStore.getDatabase().updateRecord(new User(Id = UserInfo.getUserId())); + + System.assertEquals(1, mockDatabaseInstance.callCount); + } + + @IsTest + static void it_should_support_overriding_event_bus_instance() { + MockEventBus mockEventBusInstance = new MockEventBus(); + System.assertEquals(0, mockEventBusInstance.callCount); + LoggerDataStore.setMock(mockEventBusInstance); + System.assertEquals(mockEventBusInstance, LoggerDataStore.getEventBus()); + + LoggerDataStore.getEventBus().publishRecord(new User()); // Not an actual platform event, but for mocking/test purposes, User is fine + + System.assertEquals(1, mockEventBusInstance.callCount); + } + + @IsTest + static void it_should_support_overriding_job_queue_instance() { + MockJobQueue mockJobQueueInstance = new MockJobQueue(); + System.assertEquals(0, mockJobQueueInstance.callCount); + LoggerDataStore.setMock(mockJobQueueInstance); + System.assertEquals(mockJobQueueInstance, LoggerDataStore.getJobQueue()); + + LoggerDataStore.getJobQueue().enqueueJob(new MockQueueable()); + + System.assertEquals(1, mockJobQueueInstance.callCount); + } + + private class MockDatabase extends LoggerDataStore.Database { + public Integer callCount = 0; + + public override Database.SaveResult updateRecord(SObject record) { + this.callCount++; + return LoggerMockDataCreator.createDatabaseSaveResult(true, record.Id); + } + } + + private class MockEventBus extends LoggerDataStore.EventBus { + public Integer callCount = 0; + + public override Database.SaveResult publishRecord(SObject platformEvent) { + this.callCount++; + return LoggerMockDataCreator.createDatabaseSaveResult(true, platformEvent.Id); + } + } + + private class MockJobQueue extends LoggerDataStore.JobQueue { + public Integer callCount = 0; + + public override Id enqueueJob(Queueable queueableJob) { + this.callCount++; + return LoggerMockDataCreator.createId(Schema.AsyncApexJob.SObjectType); + } + } + + private class MockQueueable implements Queueable { + public Integer callCount = 0; + public void execute(System.QueueableContext queueableContext) { + this.callCount++; + } + } +} diff --git a/nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls-meta.xml b/nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/classes/LoggerDataStore_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls new file mode 100644 index 000000000..aa8cdd7fc --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls @@ -0,0 +1,416 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @description Test class for LoggerSObjectHandler, the trigger-handler framework used by Nebula Logger. + * To demonstrate that the framework is standalone/would work with any SObject, these test methods + * all use mock implementations of the handler, using the `User` SObject + */ +@SuppressWarnings('PMD.ApexDoc, PMD.ApexAssertionsShouldIncludeMessage, PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LoggerSObjectHandler_Tests { + private static final String FAKE_PROFILE_NAME = 'Some String'; + + @IsTest + static void it_should_return_configured_sobject_handler() { + Schema.SObjectType sobjectType = new MockSObjectHandler().getSObjectType(); + LoggerSObjectHandler.setMock( + new LoggerSObjectHandler__mdt( + IsEnabled__c = true, + SObjectHandlerApexClass__c = MockSObjectHandler.class.getName(), + SObjectType__c = sobjectType.getDescribe().getName() + ) + ); + + LoggerSObjectHandler configuredInstance = LoggerSObjectHandler.getHandler(sobjectType); + + System.assertEquals( + true, + configuredInstance instanceof MockSObjectHandler, + 'The returned handler should be an instance of the configured class, MockSObjectHandler' + ); + } + + @IsTest + static void it_should_return_default_sobject_handler_implementation_when_no_configuration_provided() { + Schema.SObjectType sobjectType = new MockDefaultImplementationSObjectHandler().getSObjectType(); + LoggerSObjectHandler defaultImplementation = new MockDefaultImplementationSObjectHandler(); + + LoggerSObjectHandler configuredInstance = LoggerSObjectHandler.getHandler(sobjectType, defaultImplementation); + + System.assertEquals( + true, + configuredInstance instanceof MockDefaultImplementationSObjectHandler, + 'The returned handler should be an instance of the default implementation class, MockDefaultImplementationSObjectHandler' + ); + } + + @IsTest + static void it_should_override_triggerable_context() { + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + LoggerTriggerableContext originalContext = mockHandler.input; + LoggerTriggerableContext customContext = new LoggerTriggerableContext( + Schema.User.SObjectType, + TriggerOperation.BEFORE_INSERT, + new List{ new User(Id = UserInfo.getUserId(), Email = UserInfo.getUserEmail()) } + ); + System.assertNotEquals(originalContext, customContext); + + mockHandler.overrideTriggerableContext(customContext); + + System.assertNotEquals(originalContext, mockHandler.input); + System.assertEquals(customContext, mockHandler.input); + } + + @IsTest + static void it_should_not_run_when_disabled_via_logger_parameter() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.handlerConfiguration = new LoggerSObjectHandler__mdt(IsEnabled__c = false); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_INSERT; + mockHandler.triggerNew = records; + mockHandler.triggerNewMap = null; + mockHandler.triggerOldMap = null; + + mockHandler.execute(); + + System.assertEquals(0, mockHandler.executionCount, mockHandler); + } + + @IsTest + static void it_should_not_run_when_disabled_via_private_method() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.handlerConfiguration = new LoggerSObjectHandler__mdt(IsEnabled__c = true); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_INSERT; + mockHandler.triggerNew = records; + mockHandler.triggerNewMap = null; + mockHandler.triggerOldMap = null; + + LoggerSObjectHandler.shouldExecute(false); + mockHandler.execute(); + + System.assertEquals(0, mockHandler.executionCount, mockHandler); + } + + @IsTest + static void it_should_run_before_insert_method() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_INSERT; + mockHandler.triggerNew = records; + mockHandler.triggerNewMap = null; + mockHandler.triggerOldMap = null; + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(TriggerOperation.BEFORE_INSERT, mockHandler.executedTriggerOperationType); + System.assertEquals(mockHandler.triggerNew, mockHandler.executedTriggerNew); + System.assertEquals(null, mockHandler.executedTriggerNewMap); + System.assertEquals(null, mockHandler.executedTriggerOldMap); + } + + @IsTest + static void it_should_run_before_update_method() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_UPDATE; + mockHandler.triggerNew = null; + mockHandler.triggerNewMap = new Map(records); + mockHandler.triggerOldMap = new Map(records); + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(TriggerOperation.BEFORE_UPDATE, mockHandler.executedTriggerOperationType); + System.assertEquals(null, mockHandler.executedTriggerNew); + System.assertEquals(mockHandler.triggerNewMap, mockHandler.executedTriggerNewMap); + System.assertEquals(mockHandler.triggerOldMap, mockHandler.executedTriggerOldMap); + } + + @IsTest + static void it_should_run_before_delete_method() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_DELETE; + mockHandler.triggerNew = null; + mockHandler.triggerNewMap = new Map(records); + mockHandler.triggerOldMap = null; + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(TriggerOperation.BEFORE_DELETE, mockHandler.executedTriggerOperationType); + System.assertEquals(null, mockHandler.executedTriggerNew); + System.assertEquals(mockHandler.triggerNewMap, mockHandler.executedTriggerNewMap); + System.assertEquals(null, mockHandler.executedTriggerOldMap); + } + + @IsTest + static void it_should_run_after_insert_methods() { + // To handle AFTER_INSERT on LogEntryEvent__e, LoggerSObjectHandler has 2 methods - one with + // a list of SObject records (triggerNew), and another with a map of SObject records (triggerNewMap) + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.AFTER_INSERT; + mockHandler.triggerNew = records; + mockHandler.triggerNewMap = new Map(records); + mockHandler.triggerOldMap = null; + + mockHandler.execute(); + + System.assertEquals(2, mockHandler.executionCount); + System.assertEquals(TriggerOperation.AFTER_INSERT, mockHandler.executedTriggerOperationType); + System.assertEquals(mockHandler.triggerNew, mockHandler.executedTriggerNew); + System.assertEquals(mockHandler.triggerNewMap, mockHandler.executedTriggerNewMap); + System.assertEquals(null, mockHandler.executedTriggerOldMap); + } + + @IsTest + static void it_should_run_after_update_method() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.AFTER_UPDATE; + mockHandler.triggerNew = null; + mockHandler.triggerNewMap = new Map(records); + mockHandler.triggerOldMap = new Map(records); + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(TriggerOperation.AFTER_UPDATE, mockHandler.executedTriggerOperationType); + System.assertEquals(null, mockHandler.executedTriggerNew); + System.assertEquals(mockHandler.triggerNewMap, mockHandler.executedTriggerNewMap); + System.assertEquals(mockHandler.triggerOldMap, mockHandler.executedTriggerOldMap); + } + + @IsTest + static void it_should_run_after_delete_method() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.AFTER_DELETE; + mockHandler.triggerNew = null; + mockHandler.triggerNewMap = new Map(records); + mockHandler.triggerOldMap = null; + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(TriggerOperation.AFTER_DELETE, mockHandler.executedTriggerOperationType); + System.assertEquals(null, mockHandler.executedTriggerNew); + System.assertEquals(mockHandler.triggerNewMap, mockHandler.executedTriggerNewMap); + System.assertEquals(null, mockHandler.executedTriggerOldMap); + } + + @IsTest + static void it_should_run_after_undelete_method() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.AFTER_UNDELETE; + mockHandler.triggerNew = null; + mockHandler.triggerNewMap = new Map(records); + mockHandler.triggerOldMap = null; + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(TriggerOperation.AFTER_UNDELETE, mockHandler.executedTriggerOperationType); + System.assertEquals(null, mockHandler.executedTriggerNew); + System.assertEquals(mockHandler.triggerNewMap, mockHandler.executedTriggerNewMap); + System.assertEquals(null, mockHandler.executedTriggerOldMap); + } + + @IsTest + static void it_should_gracefully_skip_non_existent_apex_plugin() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_INSERT; + mockHandler.triggerNew = records; + mockHandler.triggerNewMap = null; + mockHandler.triggerOldMap = null; + LoggerPlugin__mdt pluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'Mock_Plugin', + IsEnabled__c = true, + SObjectHandlerApexClass__c = 'Some_Fake_Apex_Class' + ); + mockHandler.getPluginConfigurations().add(pluginConfiguration); + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(1, mockHandler.getPluginConfigurations().size(), mockHandler.getPluginConfigurations()); + System.assertEquals(0, mockHandler.getExecutedApexPlugins().size(), mockHandler.getExecutedApexPlugins()); + } + + @IsTest + static void it_should_execute_apex_plugin() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_INSERT; + mockHandler.triggerNew = records; + mockHandler.triggerNewMap = null; + mockHandler.triggerOldMap = null; + LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'ExampleApexPlugin', + IsEnabled__c = true, + SObjectHandlerApexClass__c = MockTriggerablePlugin.class.getName() + ); + mockHandler.getPluginConfigurations().add(mockPluginConfiguration); + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(1, mockHandler.getPluginConfigurations().size(), mockHandler.getPluginConfigurations()); + System.assertEquals(1, mockHandler.getExecutedApexPlugins().size(), mockHandler.getExecutedApexPlugins()); + MockTriggerablePlugin executedApexPlugin = (MockTriggerablePlugin) mockHandler.getExecutedApexPlugins().get(0); + System.assertEquals(mockPluginConfiguration, executedApexPlugin.configuration); + System.assertEquals(mockHandler.input, executedApexPlugin.input); + } + + @IsTest + static void it_should_gracefully_skip_non_existent_flow_plugin() { + User mockUser = (User) LoggerMockDataCreator.createUser(); + mockUser.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + List records = new List{ mockUser }; + MockSObjectHandler mockHandler = new MockSObjectHandler(Schema.User.SObjectType); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_INSERT; + mockHandler.triggerNew = records; + mockHandler.triggerNewMap = null; + mockHandler.triggerOldMap = null; + LoggerPlugin__mdt pluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'Mock_Plugin', + IsEnabled__c = true, + SObjectHandlerFlowName__c = 'Some_Fake_Flow' + ); + mockHandler.getPluginConfigurations().add(pluginConfiguration); + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.executionCount); + System.assertEquals(1, mockHandler.getPluginConfigurations().size(), mockHandler.getPluginConfigurations()); + System.assertEquals(0, mockHandler.getExecutedApexPlugins().size(), mockHandler.getExecutedApexPlugins()); + } + + public class MockDefaultImplementationSObjectHandler extends LoggerSObjectHandler { + public override SObjectType getSObjectType() { + return Schema.User.SObjectType; + } + } + + public class MockSObjectHandler extends LoggerSObjectHandler { + public Integer executionCount = 0; + public TriggerOperation executedTriggerOperationType; + public List executedTriggerNew; + public Map executedTriggerNewMap; + public Map executedTriggerOldMap; + + private Schema.SObjectType sobjectType; + + public MockSObjectHandler() { + this.sobjectType = Schema.User.SObjectType; + } + + public MockSObjectHandler(Schema.SObjectType sobjectType) { + this.sobjectType = sobjectType; + } + + public override SObjectType getSObjectType() { + return this.sobjectType; + } + + protected override void executeBeforeInsert(List triggerNew) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.BEFORE_INSERT; + this.executedTriggerNew = triggerNew; + super.executeBeforeInsert(triggerNew); + } + + protected override void executeBeforeUpdate(Map triggerNewMap, Map triggerOldMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.BEFORE_UPDATE; + this.executedTriggerNewMap = triggerNewMap; + this.executedTriggerOldMap = triggerOldMap; + super.executeBeforeUpdate(triggerNewMap, triggerOldMap); + } + + protected override void executeBeforeDelete(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.BEFORE_DELETE; + this.executedTriggerNewMap = triggerNewMap; + super.executeBeforeDelete(triggerNewMap); + } + + protected override void executeAfterInsert(List triggerNew) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_INSERT; + this.executedTriggerNew = triggerNew; + super.executeAfterInsert(triggerNew); + } + + protected override void executeAfterInsert(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_INSERT; + this.executedTriggerNewMap = triggerNewMap; + super.executeAfterInsert(triggerNewMap); + } + + protected override void executeAfterUpdate(Map triggerNewMap, Map triggerOldMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_UPDATE; + this.executedTriggerNewMap = triggerNewMap; + this.executedTriggerOldMap = triggerOldMap; + super.executeAfterUpdate(triggerNewMap, triggerOldMap); + } + + protected override void executeAfterDelete(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_DELETE; + this.executedTriggerNewMap = triggerNewMap; + super.executeAfterDelete(triggerNewMap); + } + + protected override void executeAfterUndelete(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_UNDELETE; + this.executedTriggerNewMap = triggerNewMap; + super.executeAfterUndelete(triggerNewMap); + } + } + + public class MockTriggerablePlugin implements LoggerPlugin.Triggerable { + public Boolean hasExecuted { get; private set; } + public LoggerPlugin__mdt configuration { get; private set; } + public LoggerTriggerableContext input { get; private set; } + + public void execute(LoggerPlugin__mdt configuration, LoggerTriggerableContext input) { + this.hasExecuted = true; + this.configuration = configuration; + this.input = input; + } + } +} diff --git a/nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls-meta.xml b/nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/classes/LoggerSObjectHandler_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls new file mode 100644 index 000000000..663397480 --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls @@ -0,0 +1,97 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @description Test class for LoggerTriggerableContext, part of the trigger-handler framework used by Nebula Logger. + * To demonstrate that the framework is standalone/would work with any SObject, these test methods + * all use mock records for the User SObject + */ +@SuppressWarnings('PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LoggerTriggerableContext_Tests { + // Based on the trigger variables provided by the platform, currently only the TriggerOperations + // BEFORE_INSERT, BEFORE_UPDATE and BEFORE_DELETE need to be tested - other operations & permutations + // will have the same parameters as one of these three scenarios + + @IsTest + static void it_constructs_instance_when_before_insert_operation() { + Schema.SObjectType sobjectType = Schema.User.SObjectType; + TriggerOperation triggerOperationType = TriggerOperation.BEFORE_INSERT; + List newUsers = new List(); + for (Integer i = 0; i < 3; i++) { + newUsers.add((User) LoggerMockDataCreator.createUser()); + } + + LoggerTriggerableContext context = new LoggerTriggerableContext(sobjectType, triggerOperationType, newUsers, null, null); + + System.assertEquals(sobjectType, context.sobjectType); + System.assertEquals(sobjectType.getDescribe().getName(), context.sobjectTypeName); + System.assertEquals(triggerOperationType, context.triggerOperationType); + System.assertEquals(newUsers, context.triggerNew); + System.assertEquals(null, context.triggerNewMap); + System.assertEquals(null, context.triggerOldMap); + System.assertEquals(newUsers.size(), context.triggerRecords.size()); + for (Integer i = 0; i < newUsers.size(); i++) { + System.assertEquals(newUsers.get(i), context.triggerRecords.get(i).triggerRecordNew); + System.assertEquals(null, context.triggerRecords.get(i).triggerRecordOld); + } + } + + @IsTest + static void it_constructs_instance_when_before_update_operation() { + Schema.SObjectType sobjectType = Schema.User.SObjectType; + TriggerOperation triggerOperationType = TriggerOperation.BEFORE_UPDATE; + List updatedUsers = new List(); + for (Integer i = 0; i < 3; i++) { + User user = (User) LoggerMockDataCreator.createUser(); + user.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + updatedUsers.add(user); + } + Map newUsersMap = new Map(updatedUsers.clone()); + Map oldUsersMap = new Map(updatedUsers.clone()); + + LoggerTriggerableContext context = new LoggerTriggerableContext(sobjectType, triggerOperationType, updatedUsers, newUsersMap, oldUsersMap); + + System.assertEquals(sobjectType, context.sobjectType); + System.assertEquals(sobjectType.getDescribe().getName(), context.sobjectTypeName); + System.assertEquals(triggerOperationType, context.triggerOperationType); + System.assertEquals(updatedUsers, context.triggerNew); + System.assertEquals(newUsersMap, context.triggerNewMap); + System.assertEquals(oldUsersMap, context.triggerOldMap); + System.assertEquals(updatedUsers.size(), context.triggerRecords.size()); + for (Integer i = 0; i < updatedUsers.size(); i++) { + User user = updatedUsers.get(i); + System.assertEquals(user, context.triggerRecords.get(i).triggerRecordNew); + System.assertEquals(oldUsersMap.get(user.Id), context.triggerRecords.get(i).triggerRecordOld); + } + } + + @IsTest + static void it_constructs_instance_when_before_delete_operation() { + Schema.SObjectType sobjectType = Schema.User.SObjectType; + TriggerOperation triggerOperationType = TriggerOperation.BEFORE_DELETE; + Map deletedLogsMap = new Map(); + for (Integer i = 0; i < 3; i++) { + User user = (User) LoggerMockDataCreator.createDataBuilder(sobjectType).populateMockId().populateRequiredFields().getRecord(); + user.Id = LoggerMockDataCreator.createId(Schema.User.SObjectType); + deletedLogsMap.put(user.Id, user); + } + + LoggerTriggerableContext context = new LoggerTriggerableContext(sobjectType, triggerOperationType, null, null, deletedLogsMap); + + System.assertEquals(sobjectType, context.sobjectType); + System.assertEquals(sobjectType.getDescribe().getName(), context.sobjectTypeName); + System.assertEquals(triggerOperationType, context.triggerOperationType); + System.assertEquals(null, context.triggerNew); + System.assertEquals(null, context.triggerNewMap); + System.assertEquals(deletedLogsMap, context.triggerOldMap); + System.assertEquals(deletedLogsMap.size(), context.triggerRecords.size()); + for (Integer i = 0; i < deletedLogsMap.size(); i++) { + User deletedLog = deletedLogsMap.values().get(i); + System.assertEquals(null, context.triggerRecords.get(i).triggerRecordNew); + System.assertEquals(deletedLog, context.triggerRecords.get(i).triggerRecordOld); + } + } +} diff --git a/nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls-meta.xml b/nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/classes/LoggerTriggerableContext_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/core/tests/logger-engine/classes/Logger_Tests.cls b/nebula-logger/core/tests/logger-engine/classes/Logger_Tests.cls index 3b8487d71..98252de77 100644 --- a/nebula-logger/core/tests/logger-engine/classes/Logger_Tests.cls +++ b/nebula-logger/core/tests/logger-engine/classes/Logger_Tests.cls @@ -3,219 +3,36 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// -@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount, PMD.NcssTypeCount') +@IsTest(IsParallel=false) private class Logger_Tests { - // Helper classes - private class SuccessCalloutMock implements HttpCalloutMock { - public HttpResponse respond(HttpRequest request) { - HttpResponse response = new HttpResponse(); - response.setBody(request.getBody()); - response.setStatusCode(200); - return response; - } - } - - private class FailureCalloutMock implements HttpCalloutMock { - public HttpResponse respond(HttpRequest request) { - HttpResponse response = new HttpResponse(); - response.setBody(request.getBody()); - response.setStatusCode(400); - return response; - } - } - - // Helper methods - static void setUserLoggingLevel(LoggingLevel loggingLevel) { - Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); - } - - static String getMessage() { - return 'Hello, world'; - } - - static LogMessage getLogMessage() { - return new LogMessage('The current user is {0}', getRecord()); - } - - static SObject getRecord() { - return new User( - Id = UserInfo.getUserId(), - ProfileId = UserInfo.getProfileId(), - Username = UserInfo.getUserName(), - UserRoleId = UserInfo.getUserRoleId() - ); - } - - static Exception getException() { - return new DmlException('Example DML Exception'); - } - - static Database.DeleteResult getDeleteResult(Boolean isSuccess) { - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - - if (isSuccess == false) { - // Delete the log twice to trigger a DeleteResult error - delete log; - } - - Database.DeleteResult deleteResult = Database.delete(log, false); - System.assertEquals(isSuccess, deleteResult.isSuccess()); - - return deleteResult; - } - - static List getDeleteResultList() { - List logs = new List(); - for (Integer i = 0; i < 3; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - insert logs; - - return Database.delete(logs, false); - } - - // TODO only some standard objects can be merged - need to find a way to either mock or trigger a merge result that works in all orgs - // static Database.MergeResult getMergeResult(Boolean isSuccess) { } - - static Database.SaveResult getSaveResult(Boolean isSuccess) { - Database.SaveResult saveResult; - - if (isSuccess == false) { - saveResult = Database.insert(new Account(), false); - } else { - saveResult = Database.insert(new Log__c(TransactionId__c = '1234'), false); - } - System.assertEquals(isSuccess, saveResult.isSuccess()); - return saveResult; - } - - static List getSaveResultList() { - List logs = new List(); - for (Integer i = 0; i < 3; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - - return Database.insert(logs, false); - } - - static Database.UpsertResult getUpsertResult(Boolean isSuccess, Boolean isInsert) { - Database.UpsertResult upsertResult; - - if (isSuccess == false && isInsert == true) { - upsertResult = Database.upsert(new Account(), false); - } else if (isSuccess == true && isInsert == true) { - upsertResult = Database.upsert(new Log__c(TransactionId__c = '1234'), false); - } else if (isSuccess == true && isInsert == false) { - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - log.TransactionId__c = '9876'; - - upsertResult = Database.upsert(log, false); - } else if (isSuccess == false && isInsert == false) { - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - - // Use a log entry this time to handle the upsert scenario - LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, Name = 'Test Log Entry'); - insert logEntry; - - // Set name field to a value too long for the field - Integer maxFieldLength = Schema.LogEntry__c.Name.getDescribe().getLength(); - logEntry.Name = '0'.repeat(maxFieldLength + 10); - - upsertResult = Database.upsert(logEntry, false); - } - System.assertEquals(isSuccess, upsertResult.isSuccess()); - System.assertEquals(isInsert, upsertResult.isCreated()); - return upsertResult; - } - - static List getUpsertResultList() { - List logs = new List(); - for (Integer i = 0; i < 3; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - - return Database.upsert(logs, false); - } - - static Database.UndeleteResult getUndeleteResult(Boolean isSuccess) { - Database.UndeleteResult undeleteResult; - - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - delete log; - - if (isSuccess == false) { - // Hard delete to cause an undelete error - Database.emptyRecycleBin(log); - - undeleteResult = Database.undelete(log.Id, false); - } else { - undeleteResult = Database.undelete(log.Id, false); - } - System.assertEquals(isSuccess, undeleteResult.isSuccess()); - return undeleteResult; - } + @IsTest + static void it_should_return_version_number() { + String expectedVersionNumber = Logger.CURRENT_VERSION_NUMBER; - static List getUndeleteResultList() { - List logs = new List(); - for (Integer i = 0; i < 3; i++) { - Log__c log = new Log__c(TransactionId__c = '1234' + i); - logs.add(log); - } - insert logs; - delete logs; + String returnedVersionNumber = Logger.getVersionNumber(); - return Database.undelete(logs, false); + System.assertEquals(expectedVersionNumber, returnedVersionNumber); } - static String getOriginLocation() { - String originLocation; - for (String currentStackTraceLine : new DmlException().getStackTraceString().split('\n')) { - if (currentStackTraceLine.contains('Logger_Tests.getOriginLocation')) { - continue; - } - if (currentStackTraceLine.contains('.LogEntryEventBuilder.')) { - continue; - } - if (currentStackTraceLine.contains('.Logger.')) { - continue; - } - originLocation = currentStackTraceLine.substringBefore(':'); - if (originLocation.startsWith('Class.')) { - originLocation = originLocation.substringAfter('Class.'); - } - break; - } + @IsTest + static void it_should_return_namespace_prefix() { + String className = Logger_Tests.class.getName(); + String expectedNamespacePrefix = className.contains('.') ? className.substringBefore('.') : ''; - return originLocation; - } + String returnedNamespacePrefix = Logger.getNamespacePrefix(); - @TestSetup - static void setup() { - LoggerSettings__c settings = LoggerSettings__c.getInstance(); - settings.IsEnabled__c = true; - settings.LoggingLevel__c = LoggingLevel.DEBUG.name(); - upsert settings; + System.assertEquals(expectedNamespacePrefix, returnedNamespacePrefix); } @IsTest static void it_should_use_in_memory_default_settings_when_not_configured() { - delete [SELECT Id FROM LoggerSettings__c]; - List existingSettings = [SELECT Id FROM LoggerSettings__c]; - System.assertEquals(0, existingSettings.size()); LoggerSettings__c expectedSettings = (LoggerSettings__c) LoggerSettings__c.SObjectType.newSObject(null, true); expectedSettings.SetupOwnerId = UserInfo.getUserId(); LoggerSettings__c returnedSettings = Logger.getUserSettings(); - existingSettings = [SELECT Id FROM LoggerSettings__c]; + List existingSettings = [SELECT Id FROM LoggerSettings__c]; System.assertEquals(0, existingSettings.size(), 'LoggerSettings__c record should not have been saved'); System.assertEquals(expectedSettings, returnedSettings); System.assertEquals(null, returnedSettings.Id); @@ -223,9 +40,6 @@ private class Logger_Tests { @IsTest static void it_should_use_org_default_settings_when_configured() { - delete [SELECT Id FROM LoggerSettings__c]; - List existingSettings = [SELECT Id FROM LoggerSettings__c]; - System.assertEquals(0, existingSettings.size()); LoggerSettings__c expectedSettings = LoggerSettings__c.getOrgDefaults(); expectedSettings.LoggingLevel__c = LoggingLevel.FINEST.name(); insert expectedSettings; @@ -235,7 +49,7 @@ private class Logger_Tests { LoggerSettings__c returnedSettings = Logger.getUserSettings(); - existingSettings = [SELECT Id FROM LoggerSettings__c]; + List existingSettings = [SELECT Id FROM LoggerSettings__c]; System.assertEquals(1, existingSettings.size(), 'LoggerSettings__c org defaults should have been saved'); System.assertEquals(expectedSettings, returnedSettings); System.assertEquals(null, returnedSettings.Id); @@ -243,9 +57,6 @@ private class Logger_Tests { @IsTest static void it_should_use_profile_settings_when_configured() { - delete [SELECT Id FROM LoggerSettings__c]; - List existingSettings = [SELECT Id FROM LoggerSettings__c]; - System.assertEquals(0, existingSettings.size()); insert LoggerSettings__c.getOrgDefaults(); LoggerSettings__c expectedSettings = LoggerSettings__c.getOrgDefaults(); expectedSettings.Id = null; @@ -258,7 +69,7 @@ private class Logger_Tests { LoggerSettings__c returnedSettings = Logger.getUserSettings(); - existingSettings = [SELECT Id FROM LoggerSettings__c]; + List existingSettings = [SELECT Id FROM LoggerSettings__c]; System.assertEquals(2, existingSettings.size(), 'LoggerSettings__c org defaults and profile settings should have been saved'); System.assertEquals(expectedSettings, returnedSettings); System.assertEquals(null, returnedSettings.Id); @@ -266,9 +77,6 @@ private class Logger_Tests { @IsTest static void it_should_use_user_settings_when_configured() { - delete [SELECT Id FROM LoggerSettings__c]; - List existingSettings = [SELECT Id FROM LoggerSettings__c]; - System.assertEquals(0, existingSettings.size()); insert LoggerSettings__c.getOrgDefaults(); LoggerSettings__c profileSettings = LoggerSettings__c.getOrgDefaults(); profileSettings.Id = null; @@ -284,7 +92,7 @@ private class Logger_Tests { LoggerSettings__c returnedSettings = Logger.getUserSettings(); - existingSettings = [SELECT Id FROM LoggerSettings__c]; + List existingSettings = [SELECT Id FROM LoggerSettings__c]; System.assertEquals(3, existingSettings.size(), 'LoggerSettings__c org defaults, profile settings, and user settings should have been saved'); System.assertEquals(expectedSettings, returnedSettings); System.assertEquals(expectedSettings.Id, returnedSettings.Id); @@ -337,7 +145,7 @@ private class Logger_Tests { } Logger.saveLog(); System.assertEquals(transactionScenarioName, Logger.getScenario()); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); Log__c log = [SELECT Id, Scenario__c FROM Log__c WHERE TransactionId__c = :transactionId]; System.assertEquals(transactionScenarioName, log.Scenario__c, 'Scenario__c was not properly set on the Log__c record'); @@ -354,7 +162,7 @@ private class Logger_Tests { } Logger.saveLog(); System.assertEquals(transactionScenarioName, Logger.getScenario()); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); Log__c log = [SELECT Id, Scenario__c FROM Log__c WHERE TransactionId__c = :transactionId]; System.assertEquals(transactionScenarioName, log.Scenario__c, 'Scenario__c was not properly set on the Log__c record'); @@ -375,7 +183,7 @@ private class Logger_Tests { } Logger.saveLog(); System.assertEquals(transactionScenarioName, Logger.getScenario()); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); Log__c log = [SELECT Id, Scenario__c FROM Log__c WHERE TransactionId__c = :transactionId]; System.assertEquals(transactionScenarioName, log.Scenario__c, 'Scenario__c was not properly set on the Log__c record'); @@ -398,7 +206,7 @@ private class Logger_Tests { Logger.setScenario(lastTransactionScenarioName); Logger.saveLog(); System.assertEquals(lastTransactionScenarioName, Logger.getScenario()); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); Log__c log = [SELECT Id, Scenario__c FROM Log__c WHERE TransactionId__c = :transactionId]; System.assertEquals(lastTransactionScenarioName, log.Scenario__c, 'Scenario__c was not properly set on the Log__c record'); @@ -644,19 +452,18 @@ private class Logger_Tests { Logger.saveLog(); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); Test.stopTest(); countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; System.assertEquals(logEntriesToCreate, countOfLogEntries); - System.assertEquals(1, Limits.getDmlStatements()); } @IsTest static void it_should_save_accurate_timestamp_when_logging_user_has_different_time_zone() { User automatedProcessUser = [SELECT Id, TimeZoneSidKey FROM User WHERE Name = 'Automated Process' AND Profile.Name = NULL]; - User testStandardUser = LoggerTestUtils.createUser(); + User testStandardUser = LoggerMockDataCreator.createUser(); // Make sure that the test user has a different time zone from the automated process user if (automatedProcessUser.TimeZoneSidKey == testStandardUser.TimeZoneSidKey) { @@ -683,7 +490,7 @@ private class Logger_Tests { System.assertEquals(1, Logger.getBufferSize()); Logger.saveLog(); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); } LogEntry__c logEntry = [SELECT Id, Timestamp__c FROM LogEntry__c]; @@ -715,7 +522,7 @@ private class Logger_Tests { Logger.getUserSettings().IsSavingEnabled__c = false; - System.assertEquals(Logger.SaveMethod.EVENT_BUS, Logger.transactionSaveMethod); + System.assertEquals(Logger.SaveMethod.EVENT_BUS, Logger.getSaveMethod()); Logger.debug('test log entry'); Logger.debug('another test log entry'); @@ -745,7 +552,7 @@ private class Logger_Tests { Logger.getUserSettings().DefaultSaveMethod__c = 'EVENT_BUS'; - System.assertEquals(Logger.SaveMethod.EVENT_BUS, Logger.transactionSaveMethod); + System.assertEquals(Logger.SaveMethod.EVENT_BUS, Logger.getSaveMethod()); Logger.debug('test log entry'); Logger.debug('another test log entry'); @@ -832,7 +639,7 @@ private class Logger_Tests { Test.startTest(); - System.assertEquals(Logger.SaveMethod.QUEUEABLE, Logger.transactionSaveMethod); + System.assertEquals(Logger.SaveMethod.QUEUEABLE, Logger.getSaveMethod()); Logger.debug('test log entry'); Logger.debug('another test log entry'); @@ -873,7 +680,7 @@ private class Logger_Tests { System.assertEquals(0, Limits.getCallouts()); System.assertEquals(0, Limits.getDmlStatements()); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); Test.stopTest(); @@ -904,7 +711,7 @@ private class Logger_Tests { System.assertEquals(0, Limits.getCallouts()); System.assertEquals(0, Limits.getDmlStatements()); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); Test.stopTest(); @@ -914,7 +721,7 @@ private class Logger_Tests { @IsTest static void it_should_save_via_rest_api_when_defaulted() { - Test.setMock(HttpCalloutMock.class, new SuccessCalloutMock()); + Test.setMock(HttpCalloutMock.class, LoggerMockDataCreator.createHttpCallout().setStatusCode(200)); Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; System.assertEquals(0, countOfLogEntries); @@ -923,7 +730,7 @@ private class Logger_Tests { Logger.getUserSettings().DefaultSaveMethod__c = 'REST'; - System.assertEquals(Logger.SaveMethod.REST, Logger.transactionSaveMethod); + System.assertEquals(Logger.SaveMethod.REST, Logger.getSaveMethod()); Logger.debug('test log entry'); Logger.debug('another test log entry'); @@ -943,7 +750,7 @@ private class Logger_Tests { @IsTest static void it_should_save_via_rest_api_when_specified_via_settings() { - Test.setMock(HttpCalloutMock.class, new SuccessCalloutMock()); + Test.setMock(HttpCalloutMock.class, LoggerMockDataCreator.createHttpCallout().setStatusCode(200)); Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; System.assertEquals(0, countOfLogEntries); @@ -968,7 +775,7 @@ private class Logger_Tests { @IsTest static void it_should_save_via_rest_api_when_specified_via_setSaveMethod() { - Test.setMock(HttpCalloutMock.class, new SuccessCalloutMock()); + Test.setMock(HttpCalloutMock.class, LoggerMockDataCreator.createHttpCallout().setStatusCode(200)); Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; System.assertEquals(0, countOfLogEntries); @@ -996,7 +803,7 @@ private class Logger_Tests { @IsTest static void it_should_throw_exception_when_save_via_rest_api_fails() { - Test.setMock(HttpCalloutMock.class, new FailureCalloutMock()); + Test.setMock(HttpCalloutMock.class, LoggerMockDataCreator.createHttpCallout().setStatusCode(400)); Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; System.assertEquals(0, countOfLogEntries); @@ -1025,89 +832,156 @@ private class Logger_Tests { @IsTest static void it_should_save_via_synchronous_dml_when_defaulted() { - Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; - System.assertEquals(0, countOfLogEntries); - - setUserLoggingLevel(LoggingLevel.DEBUG); - - Logger.getUserSettings().DefaultSaveMethod__c = 'SYNCHRONOUS_DML'; - - System.assertEquals(Logger.SaveMethod.SYNCHRONOUS_DML, Logger.transactionSaveMethod); - - Logger.debug('test log entry'); - Logger.debug('another test log entry'); - - System.assertEquals(0, Limits.getCallouts()); - + LoggerSObjectHandler.setMock( + new LoggerSObjectHandler__mdt( + DeveloperName = 'Mock_Log_Entry_Event_Config', + IsEnabled__c = true, + SObjectHandlerApexClass__c = MockSObjectHandler.class.getName(), + SObjectType__c = Schema.LogEntryEvent__e.SObjectType.getDescribe().getName() + ) + ); + Logger.getUserSettings().DefaultSaveMethod__c = Logger.SaveMethod.SYNCHRONOUS_DML.name(); + Logger.getUserSettings().LoggingLevel__c = LoggingLevel.DEBUG.name(); + System.assertEquals(LoggingLevel.DEBUG, Logger.getUserLoggingLevel()); + System.assertEquals(Logger.SaveMethod.SYNCHRONOUS_DML, Logger.getSaveMethod()); + List logEntryEvents = new List(); + logEntryEvents.add(Logger.debug('test log entry').getLogEntryEvent()); + logEntryEvents.add(Logger.debug('another test log entry').getLogEntryEvent()); Test.startTest(); - - Logger.saveLog(); System.assertEquals(0, Limits.getPublishImmediateDml()); System.assertEquals(0, Limits.getQueueableJobs()); System.assertEquals(0, Limits.getCallouts()); - System.assertNotEquals(0, Limits.getDmlStatements()); + System.assertEquals(logEntryEvents.size(), Logger.getBufferSize()); - // Requery & assert before Test.stopTest() since this is supposed to save synchronously - countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; - System.assertEquals(2, countOfLogEntries); + Logger.saveLog(); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(0, Limits.getPublishImmediateDml()); + System.assertEquals(0, Limits.getQueueableJobs()); + System.assertEquals(0, Limits.getCallouts()); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + MockSObjectHandler mockLogEntryEventBeforeInsertHandler = (MockSObjectHandler) LoggerSObjectHandler.getExecutedHandlers() + .get(Schema.LogEntryEvent__e.SObjectType) + .get(0); + // System.assertEquals(1, mockLogEntryEventBeforeInsertHandler.executionCount); + System.assertEquals(TriggerOperation.BEFORE_INSERT, mockLogEntryEventBeforeInsertHandler.executedTriggerOperationType); + System.assertEquals(logEntryEvents.size(), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.size()); + System.assertEquals(logEntryEvents.get(0), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.get(0)); + System.assertEquals(logEntryEvents.get(1), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.get(1)); + MockSObjectHandler mockLogEntryEventAfterInsertHandler = (MockSObjectHandler) LoggerSObjectHandler.getExecutedHandlers() + .get(Schema.LogEntryEvent__e.SObjectType) + .get(1); + // System.assertEquals(1, mockLogEntryEventAfterInsertHandler.executionCount); + System.assertEquals(TriggerOperation.AFTER_INSERT, mockLogEntryEventAfterInsertHandler.executedTriggerOperationType); + System.assertEquals(logEntryEvents.size(), mockLogEntryEventAfterInsertHandler.executedTriggerNew.size()); + System.assertEquals(logEntryEvents.get(0), mockLogEntryEventAfterInsertHandler.executedTriggerNew.get(0)); + System.assertEquals(logEntryEvents.get(1), mockLogEntryEventAfterInsertHandler.executedTriggerNew.get(1)); Test.stopTest(); } @IsTest static void it_should_save_via_synchronous_dml_when_specified_via_settings() { - Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; - System.assertEquals(0, countOfLogEntries); - - setUserLoggingLevel(LoggingLevel.DEBUG); - - Logger.debug('test log entry'); - Logger.debug('another test log entry'); - - System.assertEquals(0, Limits.getCallouts()); - + LoggerSObjectHandler.setMock( + new LoggerSObjectHandler__mdt( + DeveloperName = 'Mock_Log_Entry_Event_Config', + IsEnabled__c = true, + SObjectHandlerApexClass__c = MockSObjectHandler.class.getName(), + SObjectType__c = Schema.LogEntryEvent__e.SObjectType.getDescribe().getName() + ) + ); + List logEntryEvents = new List(); + logEntryEvents.add(Logger.debug('test log entry').getLogEntryEvent()); + logEntryEvents.add(Logger.debug('another test log entry').getLogEntryEvent()); Test.startTest(); - - Logger.saveLog(Logger.SaveMethod.SYNCHRONOUS_DML); System.assertEquals(0, Limits.getPublishImmediateDml()); System.assertEquals(0, Limits.getQueueableJobs()); System.assertEquals(0, Limits.getCallouts()); - System.assertNotEquals(0, Limits.getDmlStatements()); + System.assertEquals(logEntryEvents.size(), Logger.getBufferSize()); - // Requery & assert before Test.stopTest() since this is supposed to save synchronously - countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; - System.assertEquals(2, countOfLogEntries); + Logger.getUserSettings().DefaultSaveMethod__c = Logger.SaveMethod.SYNCHRONOUS_DML.name(); + System.assertEquals(Logger.SaveMethod.SYNCHRONOUS_DML, Logger.getSaveMethod()); + Logger.saveLog(); + System.assertEquals(0, Logger.getBufferSize()); + System.assertEquals(0, Limits.getPublishImmediateDml()); + System.assertEquals(0, Limits.getQueueableJobs()); + System.assertEquals(0, Limits.getCallouts()); + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + MockSObjectHandler mockLogEntryEventBeforeInsertHandler = (MockSObjectHandler) LoggerSObjectHandler.getExecutedHandlers() + .get(Schema.LogEntryEvent__e.SObjectType) + .get(0); + // System.assertEquals(1, mockLogEntryEventBeforeInsertHandler.executionCount); + System.assertEquals(TriggerOperation.BEFORE_INSERT, mockLogEntryEventBeforeInsertHandler.executedTriggerOperationType); + System.assertEquals(logEntryEvents.size(), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.size()); + System.assertEquals(logEntryEvents.get(0), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.get(0)); + System.assertEquals(logEntryEvents.get(1), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.get(1)); + MockSObjectHandler mockLogEntryEventAfterInsertHandler = (MockSObjectHandler) LoggerSObjectHandler.getExecutedHandlers() + .get(Schema.LogEntryEvent__e.SObjectType) + .get(1); + // System.assertEquals(1, mockLogEntryEventAfterInsertHandler.executionCount); + System.assertEquals(TriggerOperation.AFTER_INSERT, mockLogEntryEventAfterInsertHandler.executedTriggerOperationType); + System.assertEquals(logEntryEvents.size(), mockLogEntryEventAfterInsertHandler.executedTriggerNew.size()); + System.assertEquals(logEntryEvents.get(0), mockLogEntryEventAfterInsertHandler.executedTriggerNew.get(0)); + System.assertEquals(logEntryEvents.get(1), mockLogEntryEventAfterInsertHandler.executedTriggerNew.get(1)); Test.stopTest(); } @IsTest static void it_should_save_via_synchronous_dml_when_specified_via_setSaveMethod() { - Integer countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; - System.assertEquals(0, countOfLogEntries); - - setUserLoggingLevel(LoggingLevel.DEBUG); - - Logger.debug('test log entry'); - Logger.debug('another test log entry'); - - System.assertEquals(0, Limits.getCallouts()); - + LoggerSObjectHandler.setMock( + new LoggerSObjectHandler__mdt( + DeveloperName = 'Mock_Log_Entry_Event_Config', + IsEnabled__c = true, + SObjectHandlerApexClass__c = MockSObjectHandler.class.getName(), + SObjectType__c = Schema.LogEntryEvent__e.SObjectType.getDescribe().getName() + ) + ); + List logEntryEvents = new List(); + logEntryEvents.add(Logger.debug('test log entry').getLogEntryEvent()); + logEntryEvents.add(Logger.debug('another test log entry').getLogEntryEvent()); Test.startTest(); + System.assertEquals(0, Limits.getPublishImmediateDml()); + System.assertEquals(0, Limits.getQueueableJobs()); + System.assertEquals(0, Limits.getCallouts()); + System.assertEquals(logEntryEvents.size(), Logger.getBufferSize()); Logger.SaveMethod expectedSaveMethod = Logger.SaveMethod.SYNCHRONOUS_DML; Logger.setSaveMethod(expectedSaveMethod); System.assertEquals(expectedSaveMethod, Logger.getSaveMethod()); Logger.saveLog(); + System.assertEquals(0, Limits.getPublishImmediateDml()); System.assertEquals(0, Limits.getQueueableJobs()); System.assertEquals(0, Limits.getCallouts()); - System.assertNotEquals(0, Limits.getDmlStatements()); - - // Requery & assert before Test.stopTest() since this is supposed to save synchronously - countOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; - System.assertEquals(2, countOfLogEntries); - + System.assertEquals( + 2, + LoggerSObjectHandler.getExecutedHandlers().get(Schema.LogEntryEvent__e.SObjectType).size(), + 'Handler class should have executed two times - once for BEFORE_INSERT and once for AFTER_INSERT' + ); + MockSObjectHandler mockLogEntryEventBeforeInsertHandler = (MockSObjectHandler) LoggerSObjectHandler.getExecutedHandlers() + .get(Schema.LogEntryEvent__e.SObjectType) + .get(0); + // System.assertEquals(1, mockLogEntryEventBeforeInsertHandler.executionCount); + System.assertEquals(TriggerOperation.BEFORE_INSERT, mockLogEntryEventBeforeInsertHandler.executedTriggerOperationType); + System.assertEquals(logEntryEvents.size(), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.size()); + System.assertEquals(logEntryEvents.get(0), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.get(0)); + System.assertEquals(logEntryEvents.get(1), mockLogEntryEventBeforeInsertHandler.executedTriggerNew.get(1)); + MockSObjectHandler mockLogEntryEventAfterInsertHandler = (MockSObjectHandler) LoggerSObjectHandler.getExecutedHandlers() + .get(Schema.LogEntryEvent__e.SObjectType) + .get(1); + // System.assertEquals(1, mockLogEntryEventAfterInsertHandler.executionCount); + System.assertEquals(TriggerOperation.AFTER_INSERT, mockLogEntryEventAfterInsertHandler.executedTriggerOperationType); + System.assertEquals(logEntryEvents.size(), mockLogEntryEventAfterInsertHandler.executedTriggerNew.size()); + System.assertEquals(logEntryEvents.get(0), mockLogEntryEventAfterInsertHandler.executedTriggerNew.get(0)); + System.assertEquals(logEntryEvents.get(1), mockLogEntryEventAfterInsertHandler.executedTriggerNew.get(1)); Test.stopTest(); } @@ -1232,7 +1106,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -1256,7 +1130,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(false); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(false); Test.startTest(); @@ -1304,7 +1178,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -1328,7 +1202,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(false); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(false); Test.startTest(); @@ -1376,7 +1250,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -1400,7 +1274,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(false); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -1448,7 +1322,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -1472,7 +1346,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, true); Test.startTest(); @@ -1496,7 +1370,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -1520,7 +1394,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, false); Test.startTest(); @@ -1748,7 +1622,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -1772,7 +1646,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(false); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(false); Test.startTest(); @@ -1796,7 +1670,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -1820,7 +1694,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(false); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(false); Test.startTest(); @@ -1844,7 +1718,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -1868,7 +1742,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(false); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -1892,7 +1766,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -1916,7 +1790,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, true); Test.startTest(); @@ -1940,7 +1814,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -1964,7 +1838,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, false); Test.startTest(); @@ -2263,7 +2137,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -2287,7 +2161,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(false); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(false); Test.startTest(); @@ -2311,7 +2185,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -2335,7 +2209,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(false); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(false); Test.startTest(); @@ -2359,7 +2233,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -2383,7 +2257,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(false); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -2407,7 +2281,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -2431,7 +2305,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, true); Test.startTest(); @@ -2455,7 +2329,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -2479,7 +2353,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, false); Test.startTest(); @@ -2683,7 +2557,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -2707,7 +2581,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(false); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(false); Test.startTest(); @@ -2731,7 +2605,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -2755,7 +2629,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(false); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(false); Test.startTest(); @@ -2779,7 +2653,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -2803,7 +2677,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(false); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -2827,7 +2701,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -2851,7 +2725,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, true); Test.startTest(); @@ -2875,7 +2749,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -2899,7 +2773,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(false, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(false, false); Test.startTest(); @@ -3103,7 +2977,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -3127,7 +3001,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -3151,7 +3025,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -3175,7 +3049,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -3199,7 +3073,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -3410,7 +3284,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -3434,7 +3308,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -3458,7 +3332,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -3482,7 +3356,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -3506,7 +3380,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -3717,7 +3591,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -3741,7 +3615,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -3765,7 +3639,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -3789,7 +3663,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -3813,7 +3687,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -4024,7 +3898,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -4048,7 +3922,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -4072,7 +3946,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -4096,7 +3970,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -4120,7 +3994,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -4331,7 +4205,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -4355,7 +4229,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -4379,7 +4253,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -4403,7 +4277,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -4427,7 +4301,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -4638,7 +4512,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -4662,7 +4536,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -4686,7 +4560,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -4710,7 +4584,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -4734,7 +4608,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -4945,7 +4819,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -4969,7 +4843,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -4993,7 +4867,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -5017,7 +4891,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -5041,7 +4915,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -5252,7 +5126,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -5276,7 +5150,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -5300,7 +5174,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -5324,7 +5198,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -5348,7 +5222,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -5559,7 +5433,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -5583,7 +5457,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -5607,7 +5481,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -5631,7 +5505,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -5655,7 +5529,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -5866,7 +5740,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.DeleteResult deleteResult = getDeleteResult(true); + Database.DeleteResult deleteResult = LoggerMockDataCreator.createDatabaseDeleteResult(true); Test.startTest(); @@ -5890,7 +5764,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.SaveResult saveResult = getSaveResult(true); + Database.SaveResult saveResult = LoggerMockDataCreator.createDatabaseSaveResult(true); Test.startTest(); @@ -5914,7 +5788,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UndeleteResult undeleteResult = getUndeleteResult(true); + Database.UndeleteResult undeleteResult = LoggerMockDataCreator.createDatabaseUndeleteResult(true); Test.startTest(); @@ -5938,7 +5812,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, true); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, true); Test.startTest(); @@ -5962,7 +5836,7 @@ private class Logger_Tests { setUserLoggingLevel(loggingLevel); System.assertEquals(0, Logger.getBufferSize()); - Database.UpsertResult upsertResult = getUpsertResult(true, false); + Database.UpsertResult upsertResult = LoggerMockDataCreator.createDatabaseUpsertResult(true, false); Test.startTest(); @@ -6346,4 +6220,160 @@ private class Logger_Tests { System.assertEquals(log.Id, returnedLog.Id); } // End tests for utility methods + + // Helper methods + static void setUserLoggingLevel(LoggingLevel loggingLevel) { + Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); + } + + static String getMessage() { + return 'Hello, world'; + } + + static LogMessage getLogMessage() { + return new LogMessage('The current user is {0}', getRecord()); + } + + static SObject getRecord() { + return new User( + Id = UserInfo.getUserId(), + ProfileId = UserInfo.getProfileId(), + Username = UserInfo.getUserName(), + UserRoleId = UserInfo.getUserRoleId() + ); + } + + static Exception getException() { + return new DmlException('Example DML Exception'); + } + + static List getDeleteResultList() { + List deleteResults = new List(); + for (Integer i = 0; i < 3; i++) { + deleteResults.add(LoggerMockDataCreator.createDatabaseDeleteResult(true)); + } + return deleteResults; + } + + // TODO only some standard objects can be merged - need to find a way to either mock or trigger a merge result that works in all orgs + // static Database.MergeResult getMergeResult(Boolean isSuccess) { } + + static List getSaveResultList() { + List saveResults = new List(); + for (Integer i = 0; i < 3; i++) { + saveResults.add(LoggerMockDataCreator.createDatabaseSaveResult(true)); + } + return saveResults; + } + + static List getUpsertResultList() { + List upsertresults = new List(); + for (Integer i = 0; i < 3; i++) { + upsertresults.add(LoggerMockDataCreator.createDatabaseUpsertResult(true, true)); + } + return upsertresults; + } + + static List getUndeleteResultList() { + List undeleteResults = new List(); + for (Integer i = 0; i < 3; i++) { + undeleteResults.add(LoggerMockDataCreator.createDatabaseUndeleteResult(true)); + } + return undeleteResults; + } + + static String getOriginLocation() { + String originLocation; + for (String currentStackTraceLine : new DmlException().getStackTraceString().split('\n')) { + if (currentStackTraceLine.contains('Logger_Tests.getOriginLocation')) { + continue; + } + if (currentStackTraceLine.contains('.LogEntryEventBuilder.')) { + continue; + } + if (currentStackTraceLine.contains('.Logger.')) { + continue; + } + originLocation = currentStackTraceLine.substringBefore(':'); + if (originLocation.startsWith('Class.')) { + originLocation = originLocation.substringAfter('Class.'); + } + break; + } + + return originLocation; + } + + public class MockSObjectHandler extends LoggerSObjectHandler { + public Integer executionCount = 0; + public TriggerOperation executedTriggerOperationType; + public List executedTriggerNew; + public Map executedTriggerNewMap; + public Map executedTriggerOldMap; + + private Schema.SObjectType sobjectType; + + public MockSObjectHandler() { + this.sobjectType = Schema.LogEntryEvent__e.SObjectType; + } + + public override SObjectType getSObjectType() { + return this.sobjectType; + } + + public MockSObjectHandler setSObjectType(Schema.SObjectType sobjectType) { + this.sobjectType = sobjectType; + return this; + } + + protected override void executeBeforeInsert(List triggerNew) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.BEFORE_INSERT; + this.executedTriggerNew = triggerNew; + } + + protected override void executeBeforeUpdate(Map triggerNewMap, Map triggerOldMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.BEFORE_UPDATE; + this.executedTriggerNewMap = triggerNewMap; + this.executedTriggerOldMap = triggerOldMap; + } + + protected override void executeBeforeDelete(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.BEFORE_DELETE; + this.executedTriggerNewMap = triggerNewMap; + } + + protected override void executeAfterInsert(List triggerNew) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_INSERT; + this.executedTriggerNew = triggerNew; + } + + protected override void executeAfterInsert(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_INSERT; + this.executedTriggerNewMap = triggerNewMap; + } + + protected override void executeAfterUpdate(Map triggerNewMap, Map triggerOldMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_UPDATE; + this.executedTriggerNewMap = triggerNewMap; + this.executedTriggerOldMap = triggerOldMap; + } + + protected override void executeAfterDelete(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_DELETE; + this.executedTriggerNewMap = triggerNewMap; + } + + protected override void executeAfterUndelete(Map triggerNewMap) { + this.executionCount++; + this.executedTriggerOperationType = TriggerOperation.AFTER_UNDELETE; + this.executedTriggerNewMap = triggerNewMap; + } + } } diff --git a/nebula-logger/core/tests/logger-engine/testSuites/LoggerEngine.testSuite-meta.xml b/nebula-logger/core/tests/logger-engine/testSuites/LoggerEngine.testSuite-meta.xml index 37e1afad5..abe4a81d8 100644 --- a/nebula-logger/core/tests/logger-engine/testSuites/LoggerEngine.testSuite-meta.xml +++ b/nebula-logger/core/tests/logger-engine/testSuites/LoggerEngine.testSuite-meta.xml @@ -6,6 +6,9 @@ FlowLogger_Tests FlowRecordLogEntry_Tests LogEntryEventBuilder_Tests + LoggerDataStore_Tests + LoggerSObjectHandler_Tests + LoggerTriggerableContext_Tests Logger_Tests LogMessage_Tests diff --git a/nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls b/nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls new file mode 100644 index 000000000..0f24e5608 --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls @@ -0,0 +1,220 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Test Utilities + * @description Utility class used to mock any data-related operations for the database, event bus, and queueable jobs. + * These methods are generic, and should work in any Salesforce org. + * These methods can be used when writing Apex tests for plugins. + * @see LoggerDataStore + * @see LoggerMockDataCreator + * @see LoggerTestConfigurator + */ +@SuppressWarnings('PMD.ApexDoc, PMD.EmptyStatementBlock') +@IsTest +public without sharing class LoggerMockDataStore { + private static final Map> SOBJECT_TYPE_TO_ALL_FIELDS = new Map>(); + private static final Map> SOBJECT_TYPE_TO_REQUIRED_FIELDS = new Map>(); + private static final Map> SOBJECT_TYPE_TO_STRING_FIELDS = new Map>(); + + private static final MockDatabase MOCK_DATABASE_INSTANCE = new MockDatabase(); + private static final MockEventBus MOCK_EVENT_BUS_INSTANCE = new MockEventBus(); + private static final MockJobQueue MOCK_JOB_QUEUE_INSTANCE = new MockJobQueue(); + + public static MockDatabase getDatabase() { + return MOCK_DATABASE_INSTANCE; + } + + public static MockEventBus getEventBus() { + return MOCK_EVENT_BUS_INSTANCE; + } + + public static MockJobQueue getJobQueue() { + return MOCK_JOB_QUEUE_INSTANCE; + } + + public class MockDatabase extends LoggerDataStore.Database { + private MockDatabase() { + } + + public override List insertRecords(List records) { + List mockSaveResults = new List(); + if (records == null || records.isEmpty() == true) { + return mockSaveResults; + } + + // TODO call LoggerSObjectHandler instance using BEFORE_INSERT + // TODO Set mock ID on records + // TODO call LoggerSObjectHandler instance using AFTER_INSERT + for (SObject record : records) { + validateBeforeSave(record, true); + } + + Schema.SObjectType sobjectType = records.get(0).getSObjectType(); + + LoggerTriggerableContext beforeInsertInput = new LoggerTriggerableContext(sobjectType, TriggerOperation.BEFORE_INSERT, records); + LoggerSObjectHandler beforeInsertHandler = LoggerSObjectHandler.getHandler(sobjectType); + beforeInsertHandler?.overrideTriggerableContext(beforeInsertInput).execute(); + + LoggerTriggerableContext afterInsertInput = new LoggerTriggerableContext(sobjectType, TriggerOperation.AFTER_INSERT, records); + LoggerSObjectHandler afterInsertHandler = LoggerSObjectHandler.getHandler(sobjectType); + afterInsertHandler?.overrideTriggerableContext(afterInsertInput).execute(); + + return mockSaveResults; + } + } + + public class MockEventBus extends LoggerDataStore.EventBus { + private List publishedPlatformEvents; + private Integer publishCount; + private MockEventBus() { + this.publishedPlatformEvents = new List(); + this.publishCount = 0; + } + + public Integer getPublishCallCount() { + return this.publishCount; + } + + public List getPublishedPlatformEvents() { + return this.publishedPlatformEvents; + } + + public override Database.SaveResult publishRecord(SObject platformEvent) { + return this.publishRecords(new List{ platformEvent }).get(0); + } + + public override List publishRecords(List platformEvents) { + this.publishCount++; + this.publishedPlatformEvents.addAll(platformEvents); + + List mockSaveResults = new List(); + for (SObject platformEvent : platformEvents) { + mockSaveResults.add(validateBeforeSave(platformEvent, false)); + } + return mockSaveResults; + } + + public void deliver() { + if (this.publishedPlatformEvents.isEmpty() == true) { + return; + } + + Schema.SObjectType sobjectType = this.publishedPlatformEvents.get(0).getSObjectType(); + this.deliver(sobjectType); + } + + public void deliver(Schema.SObjectType sobjectType) { + if (sobjectType == null) { + return; + } + + this.deliver(LoggerSObjectHandler.getHandler(sobjectType)); + } + + public void deliver(LoggerSObjectHandler sobjectHandlerInstance) { + LoggerTriggerableContext platformEventsTriggerContext = new LoggerTriggerableContext( + sobjectHandlerInstance.getSObjectType(), + TriggerOperation.AFTER_INSERT, + this.publishedPlatformEvents + ); + sobjectHandlerInstance?.overrideTriggerableContext(platformEventsTriggerContext).execute(); + } + } + + public class MockJobQueue extends LoggerDataStore.JobQueue { + private List enqueuedJobs; + private Integer publishCount; + private MockJobQueue() { + this.enqueuedJobs = new List(); + } + + public List getEnqueuedJobs() { + return this.enqueuedJobs; + } + + public override Id enqueueJob(Queueable queueableJob) { + this.enqueuedJobs.add(queueableJob); + + return LoggerMockDataCreator.createId(Schema.AsyncApexJob.SObjectType); + } + + public void executeJobs() { + for (Queueable job : this.enqueuedJobs) { + job.execute(null); // TODO create mock instance for System.QueueableContext queueableContext + } + } + } + + private static Database.SaveResult validateBeforeSave(SObject record, Boolean populateMockId) { + // TODO Add checks to ensure: + // 1. All required fields are populated + // 2. All string fields are <= max field length + // 3. Any restricted picklists have a valid value set + Boolean recordIsValid = true; + loadFields(record.getSObjectType()); + validateRequiredFieldsArePopulated(record); + validateStringFieldsAreNotTooLong(record); + + if (recordIsValid == true && populateMockId == true) { + record.Id = LoggerMockDataCreator.createId(record.getSObjectType()); + } + return LoggerMockDataCreator.createDatabaseSaveResult(recordIsValid, record.Id); + } + + private static void validateRequiredFieldsArePopulated(SObject record) { + Schema.SObjectType sobjectType = record.getSObjectType(); + for (Schema.SObjectField requiredField : SOBJECT_TYPE_TO_REQUIRED_FIELDS.get(sobjectType)) { + if (requiredField.getDescribe().getSoapType() == Schema.SoapType.BOOLEAN) { + continue; + } + + Boolean recordHasField = record.getPopulatedFieldsAsMap().containsKey(requiredField.getDescribe().getName()); + Boolean recordHasFieldValue = record.get(requiredField) != null; + if (recordHasField == false || recordHasFieldValue == false) { + SObjectException ex = new SObjectException(); + ex.setMessage('The field ' + requiredField + ' is required on record: ' + JSON.serializePretty(record)); + throw ex; + } + } + } + + private static void validateStringFieldsAreNotTooLong(SObject record) { + Schema.SObjectType sobjectType = record.getSObjectType(); + for (Schema.SObjectField stringField : SOBJECT_TYPE_TO_STRING_FIELDS.get(sobjectType)) { + Boolean isFieldValueTooLong = ((String) record.get(stringField))?.length() > stringField.getDescribe().getLength(); + if (isFieldValueTooLong == true) { + SObjectException ex = new SObjectException(); + ex.setMessage('The field value for ' + stringField + ' is too long: ' + record.get(stringField)); + throw ex; + } + } + } + + private static void loadFields(Schema.SObjectType sobjectType) { + if (SOBJECT_TYPE_TO_ALL_FIELDS.containsKey(sobjectType) == true && SOBJECT_TYPE_TO_REQUIRED_FIELDS.containsKey(sobjectType) == true) { + return; + } + + SOBJECT_TYPE_TO_ALL_FIELDS.put(sobjectType, new List()); + SOBJECT_TYPE_TO_REQUIRED_FIELDS.put(sobjectType, new List()); + SOBJECT_TYPE_TO_STRING_FIELDS.put(sobjectType, new List()); + for (Schema.SObjectField field : sobjectType.getDescribe().fields.getMap().values()) { + if (field.getDescribe().isCreateable() == false) { + continue; + } + + SOBJECT_TYPE_TO_ALL_FIELDS.get(sobjectType).add(field); + if (field.getDescribe().getSoapType() == Schema.SoapType.STRING) { + SOBJECT_TYPE_TO_STRING_FIELDS.get(sobjectType).add(field); + } + + if (field.getDescribe().isNillable() == false) { + // If a field is not nillable & it is createable, then it's required + SOBJECT_TYPE_TO_REQUIRED_FIELDS.get(sobjectType).add(field); + } + } + } +} diff --git a/nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls-meta.xml b/nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/core/tests/logger-engine/utilities/LoggerMockDataStore.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/core/tests/plugin-framework/classes/LoggerSObjectHandlerPlugin_Tests.cls b/nebula-logger/core/tests/plugin-framework/classes/LoggerSObjectHandlerPlugin_Tests.cls deleted file mode 100644 index 18ca544ad..000000000 --- a/nebula-logger/core/tests/plugin-framework/classes/LoggerSObjectHandlerPlugin_Tests.cls +++ /dev/null @@ -1,43 +0,0 @@ -//------------------------------------------------------------------------------------------------// -// This file is part of the Nebula Logger project, released under the MIT License. // -// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // -//------------------------------------------------------------------------------------------------// - -@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest -private class LoggerSObjectHandlerPlugin_Tests { - private static final String PLUGIN_LOG_STATUS = 'On Hold'; - - public class ExamplePlugin extends LoggerSObjectHandlerPlugin { - public override void execute( - TriggerOperation triggerOperationType, - List triggerNew, - Map triggerNewMap, - List triggerOld, - Map triggerOldMap - ) { - switch on triggerOperationType { - when BEFORE_INSERT { - for (Log__c log : (List) triggerNew) { - log.Status__c = PLUGIN_LOG_STATUS; - } - } - } - } - } - - @IsTest - static void it_should_execute_plugin_logic() { - Log__c log = new Log__c(TransactionId__c = '1234'); - System.assertEquals(null, log.Status__c); - - Test.startTest(); - - ExamplePlugin plugin = new ExamplePlugin(); - plugin.execute(triggerOperation.BEFORE_INSERT, new List{ log }, null, null, null); - - Test.stopTest(); - - System.assertEquals(PLUGIN_LOG_STATUS, log.Status__c); - } -} diff --git a/nebula-logger/core/tests/plugin-framework/testSuites/LoggerPluginFramework.testSuite-meta.xml b/nebula-logger/core/tests/plugin-framework/testSuites/LoggerPluginFramework.testSuite-meta.xml deleted file mode 100644 index a771fcaeb..000000000 --- a/nebula-logger/core/tests/plugin-framework/testSuites/LoggerPluginFramework.testSuite-meta.xml +++ /dev/null @@ -1,5 +0,0 @@ - - - LoggerParameter_Tests - LoggerSObjectHandlerPlugin_Tests - diff --git a/nebula-logger/extra-tests/flows/LogHandler_Tests_Flow.flow-meta.xml b/nebula-logger/extra-tests/flows/LogHandler_Tests_Flow.flow-meta.xml deleted file mode 100644 index 8ea80e029..000000000 --- a/nebula-logger/extra-tests/flows/LogHandler_Tests_Flow.flow-meta.xml +++ /dev/null @@ -1,213 +0,0 @@ - - - - Log_user_after_updating - - 176 - 638 - FlowRecordLogEntry - apex - - T__record - User - - CurrentTransaction - - loggingLevelName - - INFO - - - - saveLog - - true - - - - tagsString - - flow tag - - - - flowName - - LogHandler_Tests_Flow - - - - message - - Logging user after updating - - - - record - - Get_current_user - - - true - - - Log_user_before_updating - - 176 - 278 - FlowRecordLogEntry - apex - - Update_user_s_first_name - - - T__record - User - - CurrentTransaction - - loggingLevelName - - INFO - - - - tagsString - - flow tag - - - - flowName - - LogHandler_Tests_Flow - - - - message - - Logging user before updating - - - - record - - Get_current_user - - - true - - 54.0 - - Update_user_s_first_name - - 176 - 398 - - Get_current_user.FirstName - Assign - - Logger-Flow-Test - - - - Save_current_user - - - An additional Flow used test dynamically running Flow plugins from LoggerSObjectHandler - LogHandler_Tests_Flow {!$Flow.CurrentDateTime} - - - BuilderType - - LightningFlowBuilder - - - - CanvasMode - - AUTO_LAYOUT_CANVAS - - - - OriginBuilderType - - LightningFlowBuilder - - - AutoLaunchedFlow - - Get_current_user - - 176 - 158 - false - - Log_user_before_updating - - and - - Id - EqualTo - - $User.Id - - - true - User - true - - - Save_current_user - - 176 - 518 - - Log_user_after_updating - - Get_current_user - - DefaultMode - - 50 - 0 - - Get_current_user - - - Active - - Used by test class to verify that the Flow ran successfully - ranSuccessfully - Boolean - false - false - true - - false - - - - records - SObject - true - true - false - Log__c - - - triggerOld - SObject - true - true - false - Log__c - - - triggerOperationType - String - false - true - false - - diff --git a/nebula-logger/extra-tests/flows/MockLogBatchPurgerPlugin.flow-meta.xml b/nebula-logger/extra-tests/flows/MockLogBatchPurgerPlugin.flow-meta.xml new file mode 100644 index 000000000..6a204ed9b --- /dev/null +++ b/nebula-logger/extra-tests/flows/MockLogBatchPurgerPlugin.flow-meta.xml @@ -0,0 +1,70 @@ + + + 54.0 + + Assign_example_value_0 + + 176 + 158 + + someExampleVariable + Assign + + Hello, world + + + + Example Flow used to test plugin functionality within Nebula Logger's batch cleanup job, LogBatchPurger + Mock Log Batch Purger Plugin {!$Flow.CurrentDateTime} + + + BuilderType + + LightningFlowBuilder + + + + CanvasMode + + AUTO_LAYOUT_CANVAS + + + + OriginBuilderType + + LightningFlowBuilder + + + AutoLaunchedFlow + + 50 + 0 + + Assign_example_value_0 + + + Draft + + pluginConfiguration + SObject + false + true + true + LoggerPlugin__mdt + + + pluginInput + LoggerBatchableContext + Apex + false + true + true + + + someExampleVariable + String + false + false + true + + diff --git a/nebula-logger/extra-tests/flows/MockLoggerSObjectHandlerPlugin.flow-meta.xml b/nebula-logger/extra-tests/flows/MockLoggerSObjectHandlerPlugin.flow-meta.xml new file mode 100644 index 000000000..60fc05532 --- /dev/null +++ b/nebula-logger/extra-tests/flows/MockLoggerSObjectHandlerPlugin.flow-meta.xml @@ -0,0 +1,70 @@ + + + 54.0 + + Assign_example_value_0 + + 176 + 158 + + someExampleVariable + Assign + + Hello, world + + + + Example Flow used to test plugin functionality within Nebula Logger's trigger handle framework class, LoggerSObjectHandler + Mock Logger SObject Handler Plugin {!$Flow.CurrentDateTime} + + + BuilderType + + LightningFlowBuilder + + + + CanvasMode + + AUTO_LAYOUT_CANVAS + + + + OriginBuilderType + + LightningFlowBuilder + + + AutoLaunchedFlow + + 50 + 0 + + Assign_example_value_0 + + + Active + + pluginConfiguration + SObject + false + true + true + LoggerPlugin__mdt + + + pluginInput + LoggerTriggerableContext + Apex + false + true + true + + + someExampleVariable + String + false + false + true + + diff --git a/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls b/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls new file mode 100644 index 000000000..14be25fdc --- /dev/null +++ b/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls @@ -0,0 +1,208 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') +@IsTest(IsParallel=false) +private class LogBatchPurger_Tests_Flow { + // This Flow simply returns a string variable called 'someExampleVariable' + // that's used in tests below to verify if the Flow was dynamically executed + private static final String MOCK_FLOW_NAME = 'MockLogBatchPurgerPlugin'; + private static final String MOCK_FLOW_OUTPUT_VARIABLE_NAME = 'someExampleVariable'; + private static final String MOCK_FLOW_OUTPUT_VARIABLE_VALUE = 'Hello, world'; + + private static final Integer NUMBER_OF_LOG_ENTRIES_TO_CREATE = 10; + + @TestSetup + static void setupData() { + LoggerSObjectHandler.shouldExecute(false); + LoggerSettings__c settings = Logger.getUserSettings(); + settings.IsEnabled__c = false; + settings.LoggingLevel__c = LoggingLevel.FINEST.name(); + insert settings; + + Date scheduledDeletionDate = System.today().addDays(-7); + Log__c log = new Log__c(LogRetentionDate__c = scheduledDeletionDate); + LoggerMockDataCreator.createDataBuilder(log).populateRequiredFields().getRecord(); + insert log; + + List logEntries = new List(); + for (Integer i = 0; i < NUMBER_OF_LOG_ENTRIES_TO_CREATE; i++) { + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.INFO.name()); + LoggerMockDataCreator.createDataBuilder(logEntry).populateRequiredFields().getRecord(); + logEntries.add(logEntry); + } + insert logEntries; + + LoggerTag__c tag = (LoggerTag__c) LoggerMockDataCreator.createDataBuilder(Schema.LoggerTag__c.SObjectType).populateRequiredFields().getRecord(); + insert tag; + + List logEntryTags = new List(); + for (LogEntry__c logEntry : logEntries) { + LogEntryTag__c logEntryTag = new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id); + LoggerMockDataCreator.createDataBuilder(logEntryTag).populateRequiredFields().getRecord(); + logEntryTags.add(logEntryTag); + } + insert logEntryTags; + } + + @IsTest + static void it_should_run_flow_plugin_in_start_method() { + LoggerSObjectHandler.shouldExecute(false); + LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + BatchPurgerExecutionOrder__c = 1, + BatchPurgerFlowName__c = MOCK_FLOW_NAME, + DeveloperName = 'Mock_Plugin', + IsEnabled__c = true + ); + LoggerTestConfigurator.setMock(mockPluginConfiguration); + LogBatchPurger batchJobInstance = new LogBatchPurger(); + batchJobInstance.currentSObjectType = Schema.LogEntry__c.SObjectType; + Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + + batchJobInstance.start(mockBatchableContext); + + LoggerBatchableContext expectedInput = new LoggerBatchableContext(mockBatchableContext, batchJobInstance.currentSObjectType); + System.assertEquals( + 3, + batchJobInstance.getExecutedFlowPlugins().size(), + 'The map of executed Flow plugins should have 3 keys - one for each enum value in LogBatchPurger.BatchableMethod (START, EXECUTE, and FINISH)' + ); + System.assertEquals( + 1, + batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.START).size(), + 'One Flow plugin should have run in the batch job\'s start method' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).size(), + 'No Flow plugins should have run in the batch job\'s execute method' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.FINISH).size(), + 'No Flow plugins should have run in the batch job\'s finish method' + ); + Flow.Interview flowStartPlugin = batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.START).get(0); + LoggerPlugin__mdt returnedPluginConfiguration = (LoggerPlugin__mdt) flowStartPlugin.getVariableValue('pluginConfiguration'); + System.assertEquals(mockPluginConfiguration, returnedPluginConfiguration, 'LoggerPlugin__mdt records should match'); + LoggerBatchableContext returnedPluginInput = (LoggerBatchableContext) flowStartPlugin.getVariableValue('pluginInput'); + System.assertEquals(JSON.serialize(expectedInput), JSON.serialize(returnedPluginInput), 'Plugin inputs should match'); + System.assertEquals( + MOCK_FLOW_OUTPUT_VARIABLE_VALUE, + flowStartPlugin.getVariableValue(MOCK_FLOW_OUTPUT_VARIABLE_NAME), + 'Flow should have returned the variable ' + + MOCK_FLOW_OUTPUT_VARIABLE_NAME + + ' with the value + ' + + MOCK_FLOW_OUTPUT_VARIABLE_VALUE + ); + } + + @IsTest + static void it_should_run_flow_plugin_in_execute_method() { + LoggerSObjectHandler.shouldExecute(false); + LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + BatchPurgerExecutionOrder__c = 1, + BatchPurgerFlowName__c = MOCK_FLOW_NAME, + DeveloperName = 'Mock_Plugin', + IsEnabled__c = true + ); + LoggerTestConfigurator.setMock(mockPluginConfiguration); + LogBatchPurger batchJobInstance = new LogBatchPurger(); + List logsToDelete = [SELECT Id FROM Log__c]; + System.assertNotEquals(0, logsToDelete.size()); + Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + + batchJobInstance.execute(mockBatchableContext, logsToDelete); + + LoggerBatchableContext expectedInput = new LoggerBatchableContext(mockBatchableContext, batchJobInstance.currentSObjectType); + System.assertEquals( + 3, + batchJobInstance.getExecutedFlowPlugins().size(), + 'The map of executed Flow plugins should have 3 keys - one for each enum value in LogBatchPurger.BatchableMethod (START, EXECUTE, and FINISH)' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.START).size(), + 'No Flow plugins should have run in the batch job\'s start method' + ); + System.assertEquals( + 1, + batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).size(), + 'One Flow plugin should have run in the batch job\'s execute method' + ); + System.assertEquals( + 0, + batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.FINISH).size(), + 'No Flow plugins should have run in the batch job\'s finish method' + ); + Flow.Interview flowExecutePlugin = batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).get(0); + LoggerPlugin__mdt returnedPluginConfiguration = (LoggerPlugin__mdt) flowExecutePlugin.getVariableValue('pluginConfiguration'); + System.assertEquals(mockPluginConfiguration, returnedPluginConfiguration, 'LoggerPlugin__mdt records should match'); + LoggerBatchableContext returnedPluginInput = (LoggerBatchableContext) flowExecutePlugin.getVariableValue('pluginInput'); + System.assertEquals(JSON.serialize(expectedInput), JSON.serialize(returnedPluginInput), 'Plugin inputs should match'); + System.assertEquals( + MOCK_FLOW_OUTPUT_VARIABLE_VALUE, + flowExecutePlugin.getVariableValue(MOCK_FLOW_OUTPUT_VARIABLE_NAME), + 'Flow should have returned the variable ' + + MOCK_FLOW_OUTPUT_VARIABLE_NAME + + ' with the value + ' + + MOCK_FLOW_OUTPUT_VARIABLE_VALUE + ); + } + + // FIXME this method consistently fails inconsistently, sometimes with a GACK, but appears to work when manually testing it. I'll revisit it a future release. + // @IsTest + // static void it_should_run_flow_plugin_in_finish_method() { + // LoggerSObjectHandler.shouldExecute(false); + // LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + // BatchPurgerExecutionOrder__c = 1, + // BatchPurgerFlowName__c = MOCK_FLOW_NAME, + // DeveloperName = 'Mock_Plugin', + // IsEnabled__c = true + // ); + // LoggerTestConfigurator.setMock(mockPluginConfiguration); + // LogBatchPurger batchJobInstance = new LogBatchPurger(); + // List logsToDelete = [SELECT Id FROM Log__c]; + // System.assertNotEquals(0, logsToDelete.size(), 'There should be some records to delete'); + // Database.BatchableContext mockBatchableContext = LoggerMockDataCreator.createBatchableContext('some_fake_job_id'); + + // batchJobInstance.finish(mockBatchableContext); + + // LoggerBatchableContext expectedInput = new LoggerBatchableContext(mockBatchableContext, batchJobInstance.currentSObjectType); + // System.assertEquals( + // 3, + // batchJobInstance.getExecutedFlowPlugins().size(), + // 'The map of executed Flow plugins should have 3 keys - one for each enum value in LogBatchPurger.BatchableMethod (START, EXECUTE, and FINISH)' + // ); + // System.assertEquals( + // 0, + // batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.START).size(), + // 'No Flow plugins should have run in the batch job\'s start method' + // ); + // System.assertEquals( + // 0, + // batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.EXECUTE).size(), + // 'No Flow plugins should have run in the batch job\'s execute method' + // ); + // System.assertEquals( + // 1, + // batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.FINISH).size(), + // 'One Flow plugin should have run in the batch job\'s finish method' + // ); + // Flow.Interview flowFinishPlugin = batchJobInstance.getExecutedFlowPlugins().get(LogBatchPurger.BatchableMethod.FINISH).get(0); + // LoggerPlugin__mdt returnedPluginConfiguration = (LoggerPlugin__mdt) flowFinishPlugin.getVariableValue('pluginConfiguration'); + // System.assertEquals(mockPluginConfiguration, returnedPluginConfiguration, 'LoggerPlugin__mdt records should match'); + // LoggerBatchableContext returnedPluginInput = (LoggerBatchableContext) flowFinishPlugin.getVariableValue('pluginInput'); + // System.assertEquals(JSON.serialize(expectedInput), JSON.serialize(returnedPluginInput), 'Plugin inputs should match'); + // System.assertEquals( + // MOCK_FLOW_OUTPUT_VARIABLE_VALUE, + // flowFinishPlugin.getVariableValue(MOCK_FLOW_OUTPUT_VARIABLE_NAME), + // 'Flow should have returned the variable ' + + // MOCK_FLOW_OUTPUT_VARIABLE_NAME + + // ' with the value + ' + + // MOCK_FLOW_OUTPUT_VARIABLE_VALUE + // ); + // } +} diff --git a/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls-meta.xml b/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Flow.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Integration.cls b/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Integration.cls index 459cfa80e..e505762a9 100644 --- a/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Integration.cls +++ b/nebula-logger/extra-tests/tests/LogBatchPurger_Tests_Integration.cls @@ -43,7 +43,7 @@ private class LogBatchPurger_Tests_Integration { static void it_should_throw_exception_when_user_does_not_have_delete_access() { Integer originalCountOfLogEntries = [SELECT COUNT() FROM LogEntry__c]; - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); System.runAs(standardUser) { System.assertEquals(false, Schema.Log__c.SObjectType.getDescribe().isDeletable()); diff --git a/nebula-logger/extra-tests/tests/LogEntryEventBuilder_Tests_Integration.cls b/nebula-logger/extra-tests/tests/LogEntryEventBuilder_Tests_Integration.cls index a01cf8ed1..9484f4bf7 100644 --- a/nebula-logger/extra-tests/tests/LogEntryEventBuilder_Tests_Integration.cls +++ b/nebula-logger/extra-tests/tests/LogEntryEventBuilder_Tests_Integration.cls @@ -4,20 +4,17 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=false) private class LogEntryEventBuilder_Tests_Integration { private static final Profile STANDARD_USER_PROFILE = [SELECT Id FROM Profile WHERE Name IN ('Standard User', 'Usuario estándar')]; @IsTest static void stripInaccessibleFieldsForRecordWhenEnabled() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); - AccountBrand mockAccountBrand = new AccountBrand( - Id = LoggerTestUtils.createMockId(Schema.AccountBrand.SObjectType), - CompanyName = 'Some Company, Inc.', - Email = 'some.one@some.company.com', - Name = 'Something', - Phone = '510-555-1234' - ); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); + AccountBrand mockAccountBrand = (AccountBrand) LoggerMockDataCreator.createDataBuilder(Schema.AccountBrand.SObjectType) + .populateMockId() + .populateRequiredFields() + .getRecord(); // The 'standard user' profile doesn't have access to AccountBrand, // so stripAccessible will remove everything except the ID field AccountBrand strippedAccountBrand = new AccountBrand(Id = mockAccountBrand.Id); @@ -44,17 +41,14 @@ private class LogEntryEventBuilder_Tests_Integration { @IsTest static void stripInaccessibleFieldsForRecordsWhenEnabled() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); List mockAccountBrands = new List(); List strippedAccountBrands = new List(); for (Integer i = 0; i < 5; i++) { - AccountBrand mockAccountBrand = new AccountBrand( - Id = LoggerTestUtils.createMockId(Schema.AccountBrand.SObjectType), - CompanyName = 'Some Company, Inc.', - Email = 'some.one.number_' + i + '@some.company.com', - Name = 'Something', - Phone = '510-555-1234' - ); + AccountBrand mockAccountBrand = (AccountBrand) LoggerMockDataCreator.createDataBuilder(Schema.AccountBrand.SObjectType) + .populateMockId() + .populateRequiredFields() + .getRecord(); mockAccountBrands.add(mockAccountBrand); // The 'standard user' profile doesn't have access to AccountBrand, diff --git a/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Flow.cls b/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Flow.cls index b94531ded..bfa888bdb 100644 --- a/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Flow.cls +++ b/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Flow.cls @@ -4,7 +4,7 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class LogEntryHandler_Tests_Flow { private static final String EXAMPLE_FLOW_API_NAME = 'LogEntryHandler_Tests_Flow'; @@ -12,10 +12,12 @@ private class LogEntryHandler_Tests_Flow { static void setSkipSettingFlowDetailsWhenOriginLocationIsNull() { Log__c log = new Log__c(TransactionId__c = '1234'); insert log; + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, OriginLocation__c = null, OriginType__c = 'Flow'); + insert logEntry; - logEntry = getLogEntry(); + logEntry = getLogEntry(); System.assertEquals(null, logEntry.OriginLocation__c, 'Origin Location was not null.'); System.assertEquals('Flow', logEntry.OriginType__c, 'Origin Type was not equal to Flow.'); System.assertEquals(null, logEntry.FlowActiveVersionId__c, 'FlowActiveVersionId was not null.'); @@ -35,13 +37,14 @@ private class LogEntryHandler_Tests_Flow { static void setFlowDetails() { FlowDefinitionView flowDefinition = getFlowDefinition(); FlowVersionView flowVersion = getFlowVersion(flowDefinition.ActiveVersionId); - Log__c log = new Log__c(TransactionId__c = '1234'); insert log; + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, OriginLocation__c = flowDefinition.ApiName, OriginType__c = 'Flow'); + insert logEntry; - logEntry = getLogEntry(); + logEntry = getLogEntry(); System.assertEquals('Flow', logEntry.OriginType__c, 'OriginType was not flow.'); System.assertEquals(flowDefinition.ActiveVersionId, logEntry.FlowActiveVersionId__c, 'FlowActiveVersionId was incorrect.'); System.assertEquals(flowDefinition.Description, logEntry.FlowDescription__c, 'FlowDescription was incorrect.'); diff --git a/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls b/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls new file mode 100644 index 000000000..a817f03f9 --- /dev/null +++ b/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls @@ -0,0 +1,33 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') +@IsTest(IsParallel=true) +private class LogEntryHandler_Tests_Integration { + @IsTest + static void it_should_populate_related_record_name_field_on_log_entry_with_email_message_subject() { + System.assertEquals( + false, + Schema.EmailMessage.Subject.getDescribe().isNameField(), + 'This test assumes that EmailMessage does not use Subject as the object\'s display-name field' + ); + Case cas = (Case) LoggerMockDataCreator.createDataBuilder(Schema.Case.SObjectType).populateRequiredFields().getRecord(); + insert cas; + EmailMessage emailMessage = (EmailMessage) LoggerMockDataCreator.createDataBuilder(new EmailMessage(Subject = 'Some subject', ParentId = cas.Id)) + .populateRequiredFields() + .getRecord(); + insert emailMessage; + Log__c log = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + insert log; + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, RecordId__c = emailMessage.Id); + + insert logEntry; + + logEntry = [SELECT Id, RecordId__c, RecordName__c FROM LogEntry__c WHERE Id = :logEntry.Id]; + System.assertEquals(emailMessage.Id, logEntry.RecordId__c); + System.assertEquals(emailMessage.Subject, logEntry.RecordName__c); + } +} diff --git a/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls-meta.xml b/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/extra-tests/tests/LogEntryHandler_Tests_Integration.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/extra-tests/tests/LogHandler_Tests_Flow.cls b/nebula-logger/extra-tests/tests/LogHandler_Tests_Flow.cls deleted file mode 100644 index 5d388ed32..000000000 --- a/nebula-logger/extra-tests/tests/LogHandler_Tests_Flow.cls +++ /dev/null @@ -1,33 +0,0 @@ -//------------------------------------------------------------------------------------------------// -// This file is part of the Nebula Logger project, released under the MIT License. // -// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // -//------------------------------------------------------------------------------------------------// - -@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest -private class LogHandler_Tests_Flow { - @IsTest - static void runFlowPluginWhenConfigured() { - // Assumption: the Flow LogHandler_Tests_Flow makes an update to the current user's FirstName - // The specific action within the Flow isn't that important - we just want to make sure - // that that Flow is dynamically executed - String pluginFlowApiName = 'LogHandler_Tests_Flow'; - String expectedUserFirstName = 'Logger-Flow-Test'; - System.assertNotEquals(expectedUserFirstName, UserInfo.getFirstName(), 'Current user first name did not match expected user first name.'); - - Test.startTest(); - - // Use the mock configurations - LoggerPlugin__mdt plugin = new LoggerPlugin__mdt(PluginType__c = 'Flow', PluginApiName__c = pluginFlowApiName); - LoggerSObjectHandler.setMockPlugin(Schema.Log__c.SObjectType, plugin); - - Log__c log = new Log__c(TransactionId__c = '1234'); - insert log; - - Test.stopTest(); - - // Verify that the Flow ran by checking if the user's FirstName was updated - User currentUser = [SELECT Id, FirstName FROM User WHERE Id = :UserInfo.getUserId()]; - System.assertEquals(expectedUserFirstName, currentUser.FirstName, 'Current user FirstName did not match expected user first name.'); - } -} diff --git a/nebula-logger/extra-tests/tests/LogMassDeleteExtension_Tests_Integration.cls b/nebula-logger/extra-tests/tests/LogMassDeleteExtension_Tests_Integration.cls index ddc5fb085..0087b86c7 100644 --- a/nebula-logger/extra-tests/tests/LogMassDeleteExtension_Tests_Integration.cls +++ b/nebula-logger/extra-tests/tests/LogMassDeleteExtension_Tests_Integration.cls @@ -4,12 +4,14 @@ //------------------------------------------------------------------------------------------------// @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=false) private class LogMassDeleteExtension_Tests_Integration { private static final Profile STANDARD_USER_PROFILE = [SELECT Id FROM Profile WHERE Name IN ('Standard User', 'Usuario estándar')]; @TestSetup static void setupData() { + LoggerSObjectHandler.shouldExecute(false); + List logs = new List(); for (Integer i = 0; i < 10; i++) { Log__c log = new Log__c(TransactionId__c = 'TXN-' + i); @@ -31,21 +33,17 @@ private class LogMassDeleteExtension_Tests_Integration { logsToKeep.add(logs.get(i)); } } - ApexPages.StandardSetController controller = new ApexPages.StandardSetController(logs); controller.setSelected(logsToDelete); - PageReference pageReference = Page.LogMassDelete; Test.setCurrentPage(pageReference); - - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); System.runAs(standardUser) { System.assertEquals(false, Schema.Log__c.SObjectType.getDescribe().isDeletable()); - String deleteAccessError = 'You do not have access to delete logs records'; - new LogMassDeleteExtension(controller); + String deleteAccessError = 'You do not have access to delete logs records'; System.assertEquals(true, ApexPages.hasMessages(ApexPages.SEVERITY.ERROR)); System.assertEquals(deleteAccessError, ApexPages.getMessages().get(0).getSummary()); } diff --git a/nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls b/nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls new file mode 100644 index 000000000..3855b4ada --- /dev/null +++ b/nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls @@ -0,0 +1,43 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') +@IsTest(IsParallel=true) +private class LoggerSObjectHandler_Tests_Flow { + // This Flow simply returns a string variable called 'someExampleVariable' + // that's used in tests below to verify if the Flow was dynamically executed + private static final String MOCK_FLOW_NAME = 'MockLoggerSObjectHandlerPlugin'; + private static final String MOCK_FLOW_OUTPUT_VARIABLE_VALUE = 'Hello, world'; + + @IsTest + static void it_should_execute_flow_plugin() { + LoggerPlugin__mdt mockPluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'ExampleFlowPlugin', + IsEnabled__c = true, + SObjectHandlerFlowName__c = MOCK_FLOW_NAME + ); + LoggerTestConfigurator.setMock(mockPluginConfiguration); + MockLogHandler mockHandler = new MockLogHandler(); + mockHandler.triggerOperationType = TriggerOperation.BEFORE_INSERT; + mockHandler.triggerNew = new List{ new Log__c(TransactionId__c = '1234') }; + mockHandler.triggerNewMap = null; + mockHandler.triggerOldMap = null; + + mockHandler.execute(); + + System.assertEquals(1, mockHandler.getPluginConfigurations().size(), mockHandler.getPluginConfigurations()); + System.assertEquals(1, mockHandler.getExecutedFlowPlugins().size(), mockHandler.getExecutedFlowPlugins()); + Flow.Interview executedFlowPlugin = mockHandler.getExecutedFlowPlugins().get(0); + System.assertEquals(mockPluginConfiguration, executedFlowPlugin.getVariableValue('pluginConfiguration')); + System.assertEquals(mockHandler.input, executedFlowPlugin.getVariableValue('pluginInput')); + System.assertEquals(MOCK_FLOW_OUTPUT_VARIABLE_VALUE, executedFlowPlugin.getVariableValue('someExampleVariable')); + } + + private class MockLogHandler extends LoggerSObjectHandler { + public override SObjectType getSObjectType() { + return Schema.Log__c.SObjectType; + } + } +} diff --git a/nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls-meta.xml b/nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/extra-tests/tests/LoggerSObjectHandler_Tests_Flow.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/extra-tests/tests/LoggerSettingsController_Tests_Security.cls b/nebula-logger/extra-tests/tests/LoggerSettingsController_Tests_Security.cls index 78b904890..7396961a2 100644 --- a/nebula-logger/extra-tests/tests/LoggerSettingsController_Tests_Security.cls +++ b/nebula-logger/extra-tests/tests/LoggerSettingsController_Tests_Security.cls @@ -10,14 +10,14 @@ * ...from within the normal tests, and each org might have additional required fields, VR, etc on the std objects */ @SuppressWarnings('PMD.ApexDoc, PMD.MethodNamingConventions') -@IsTest +@IsTest(IsParallel=false) private class LoggerSettingsController_Tests_Security { private static final String CAN_MODIFY_LOGGER_SETTINGS_PERMISSION_NAME = 'CanModifyLoggerSettings'; private static final Profile STANDARD_USER_PROFILE = [SELECT Id FROM Profile WHERE Name IN ('Standard User', 'Usuario estándar')]; @IsTest static void it_should_permit_user_to_modify_logger_settings_when_custom_permission_is_assigned() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); insert standardUser; PermissionSet permissionSet = new PermissionSet(Name = 'CustomPermissionEnabled', Label = 'Custom Permisison Enabled'); insert permissionSet; @@ -37,9 +37,9 @@ private class LoggerSettingsController_Tests_Security { @IsTest static void it_should_permit_user_to_modify_logger_settings_when_loggerAdmin_permission_set_is_assigned() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); insert standardUser; - LoggerTestUtils.assignAdminPermissionSet(standardUser.Id); + LoggerTestConfigurator.assignAdminPermissionSet(standardUser.Id); System.runAs(standardUser) { System.assertEquals(true, FeatureManagement.checkPermission(CAN_MODIFY_LOGGER_SETTINGS_PERMISSION_NAME)); @@ -49,9 +49,9 @@ private class LoggerSettingsController_Tests_Security { @IsTest static void it_should_not_permit_user_to_modify_logger_settings_when_loggerLogViewer_permission_set_is_assigned() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); insert standardUser; - LoggerTestUtils.assignLogViewerPermissionSet(standardUser.Id); + LoggerTestConfigurator.assignLogViewerPermissionSet(standardUser.Id); System.runAs(standardUser) { System.assertEquals(false, FeatureManagement.checkPermission(CAN_MODIFY_LOGGER_SETTINGS_PERMISSION_NAME)); @@ -61,9 +61,9 @@ private class LoggerSettingsController_Tests_Security { @IsTest static void it_should_not_permit_user_to_modify_logger_settings_when_loggerEndUser_permission_set_is_assigned() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); insert standardUser; - LoggerTestUtils.assignEndUserPermissionSet(standardUser.Id); + LoggerTestConfigurator.assignEndUserPermissionSet(standardUser.Id); System.runAs(standardUser) { System.assertEquals(false, FeatureManagement.checkPermission(CAN_MODIFY_LOGGER_SETTINGS_PERMISSION_NAME)); @@ -73,9 +73,9 @@ private class LoggerSettingsController_Tests_Security { @IsTest static void it_should_not_permit_user_to_modify_logger_settings_when_loggerLogCreator_permission_set_is_assigned() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); insert standardUser; - LoggerTestUtils.assignLogCreatorPermissionSet(standardUser.Id); + LoggerTestConfigurator.assignLogCreatorPermissionSet(standardUser.Id); System.runAs(standardUser) { System.assertEquals(false, FeatureManagement.checkPermission(CAN_MODIFY_LOGGER_SETTINGS_PERMISSION_NAME)); @@ -85,7 +85,7 @@ private class LoggerSettingsController_Tests_Security { @IsTest static void it_should_not_permit_user_to_modify_logger_settings_when_custom_permission_is_not_assigned() { - User standardUser = LoggerTestUtils.createUser(STANDARD_USER_PROFILE.Id); + User standardUser = LoggerMockDataCreator.createUser(STANDARD_USER_PROFILE.Id); System.runAs(standardUser) { System.assertEquals(false, LoggerSettingsController.canUserModifyLoggerSettings()); } diff --git a/nebula-logger/extra-tests/tests/Logger_Tests_ExperienceSite.cls b/nebula-logger/extra-tests/tests/Logger_Tests_ExperienceSite.cls index 5c5e907ae..70fde7ed5 100644 --- a/nebula-logger/extra-tests/tests/Logger_Tests_ExperienceSite.cls +++ b/nebula-logger/extra-tests/tests/Logger_Tests_ExperienceSite.cls @@ -8,7 +8,7 @@ * @description Additional integration tests for orgs with Experience Sites (Communities) enabled */ @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=false) private class Logger_Tests_ExperienceSite { private static final Boolean IS_EXPERIENCE_CLOUD_ENABLED = Schema.getGlobalDescribe().containsKey('Network'); private static final String GUEST_USER_PROFILE_NAME = 'Logger Test Site Profile'; @@ -36,7 +36,7 @@ private class Logger_Tests_ExperienceSite { } User guestUser = guestUsers.get(0); - LoggerTestUtils.assignLogCreatorPermissionSet(guestUser.Id); + LoggerTestConfigurator.assignLogCreatorPermissionSet(guestUser.Id); } @IsTest @@ -92,7 +92,7 @@ private class Logger_Tests_ExperienceSite { Logger.debug(message); Logger.saveLog(); - Test.getEventBus().deliver(); + System.Test.getEventBus().deliver(); } LogEntry__c logEntry = [SELECT Id, Log__r.LoggedBy__c, Log__r.OwnerId, Log__r.UserLicenseDefinitionKey__c, Message__c FROM LogEntry__c]; diff --git a/nebula-logger/extra-tests/tests/Logger_Tests_InboundEmailHandler.cls b/nebula-logger/extra-tests/tests/Logger_Tests_InboundEmailHandler.cls index 73094cb5d..5a7b8ba47 100644 --- a/nebula-logger/extra-tests/tests/Logger_Tests_InboundEmailHandler.cls +++ b/nebula-logger/extra-tests/tests/Logger_Tests_InboundEmailHandler.cls @@ -8,35 +8,29 @@ * @description Additional integration tests for logging from an instance of `Messaging.InboundEmailHandler` */ @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class Logger_Tests_InboundEmailHandler { @IsTest - private static void saveLogsWhenRunningEmailService() { - // Create a new email and envelope object - Messaging.InboundEnvelope envelope = new Messaging.InboundEnvelope(); - - // Create the email body + private static void it_should_save_logs_when_running_within_email_service() { Messaging.InboundEmail email = new Messaging.InboundEmail(); email.plainTextBody = 'Example email content'; email.fromAddress = 'test@test.com'; email.subject = 'My example email'; - - // Create an instance of the example handler class - ExampleInboundEmailHandler handler = new ExampleInboundEmailHandler(); - + Messaging.InboundEnvelope envelope = new Messaging.InboundEnvelope(); + ExampleInboundEmailHandler emailHandler = new ExampleInboundEmailHandler(); Test.startTest(); - + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); String transactionId = Logger.getTransactionId(); - Messaging.InboundEmailResult result = handler.handleInboundEmail(email, envelope); - - Test.stopTest(); + System.assertNotEquals(1, Logger.saveLogCallCount, 'No logging should have occurred yet'); - System.assert(result.success, 'InboundEmailResult returned a failure message'); + Messaging.InboundEmailResult result = emailHandler.handleInboundEmail(email, envelope); + System.assertNotEquals(0, Logger.saveLogCallCount, 'ExampleInboundEmailHandler class should have logged & saved some logging data'); + Test.stopTest(); + System.assertEquals(true, result.success, 'InboundEmailResult returned a failure message'); List logs = [SELECT Id, TransactionId__c FROM Log__c]; System.assertEquals(1, logs.size(), 'Logs size did not match expected value of 1.'); System.assertEquals(transactionId, logs.get(0).TransactionId__c, 'Transaction Id does match expected value.'); - List logEntries = [SELECT Id, Message__c FROM LogEntry__c]; System.assertEquals(1, logEntries.size(), 'Log entries size did not match expected value of 1.'); System.assertEquals(ExampleInboundEmailHandler.logEntryMessage, logEntries.get(0).Message__c, 'Log entries message did not match expected value.'); diff --git a/nebula-logger/extra-tests/tests/Logger_Tests_MergeResult.cls b/nebula-logger/extra-tests/tests/Logger_Tests_MergeResult.cls index 2eac3d11c..32240a0d5 100644 --- a/nebula-logger/extra-tests/tests/Logger_Tests_MergeResult.cls +++ b/nebula-logger/extra-tests/tests/Logger_Tests_MergeResult.cls @@ -3,6 +3,8 @@ // See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // //------------------------------------------------------------------------------------------------// +// TODO move these test methods back into Logger_Tests, now that mocking is possible via LoggerMockDataCreator.createDatabaseMergeResult() +// (and then delete this class) /** * @group Extra Tests * @description Additional integration tests for merging records. @@ -10,7 +12,7 @@ * from within the normal tests, and each org might have additional required fields, VR, etc on the std objects */ @SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest +@IsTest(IsParallel=true) private class Logger_Tests_MergeResult { static String getMessage() { return 'Hello, world'; @@ -20,18 +22,8 @@ private class Logger_Tests_MergeResult { return new LogMessage('The current date is {0}', System.today()); } - static Database.MergeResult getMergeResult() { - Account mainAccount = new Account(Name = 'Main'); - Account duplicateAccount = new Account(Name = 'Duplicate'); - List accounts = new List{ mainAccount, duplicateAccount }; - insert accounts; - - Database.MergeResult mergeResult = Database.merge(mainAccount, duplicateAccount); - return mergeResult; - } - static List getMergeResultList() { - return new List{ getMergeResult() }; + return new List{ LoggerMockDataCreator.createDatabaseMergeResult(true) }; } @IsTest @@ -40,7 +32,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -109,7 +101,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -183,7 +175,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -252,7 +244,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -326,7 +318,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -395,7 +387,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -469,7 +461,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -538,7 +530,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -612,7 +604,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -681,7 +673,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -755,7 +747,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -824,7 +816,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -898,7 +890,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); @@ -967,7 +959,7 @@ private class Logger_Tests_MergeResult { Logger.getUserSettings().LoggingLevel__c = loggingLevel.name(); System.assertEquals(0, Logger.getBufferSize(), 'Logger buffer size was incorrect.'); - Database.MergeResult mergeResult = getMergeResult(); + Database.MergeResult mergeResult = LoggerMockDataCreator.createDatabaseMergeResult(true); Test.startTest(); diff --git a/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableStatusApiCallout__c.field-meta.xml b/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableStatusApiCallout__c.field-meta.xml index 2cbb3cce7..97fecdbad 100644 --- a/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableStatusApiCallout__c.field-meta.xml +++ b/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableStatusApiCallout__c.field-meta.xml @@ -1,6 +1,7 @@ EnableStatusApiCallout__c + DeprecateCandidate true When enabled, an async callout is made to https://api.status.salesforce.com to get additional details about the current org, including the org's release version and release number. This information is then stored on the Log__c record. diff --git a/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableSystemMessages__c.field-meta.xml b/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableSystemMessages__c.field-meta.xml index 4af1e71e9..fc82cc8ef 100644 --- a/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableSystemMessages__c.field-meta.xml +++ b/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/EnableSystemMessages__c.field-meta.xml @@ -1,6 +1,7 @@ EnableSystemMessages__c + DeprecateCandidate true When enabled, log entries may be generated that contain additional details about the logging system. false diff --git a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/IsPlatformEventStorageEnabled__c.field-meta.xml b/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/IsPlatformEventStorageEnabled__c.field-meta.xml similarity index 81% rename from nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/IsPlatformEventStorageEnabled__c.field-meta.xml rename to nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/IsPlatformEventStorageEnabled__c.field-meta.xml index ab0b46b87..4ef7fb06d 100644 --- a/nebula-logger/core/main/configuration/objects/LoggerSettings__c/fields/IsPlatformEventStorageEnabled__c.field-meta.xml +++ b/nebula-logger/managed-package/core/main/configuration/objects/LoggerSettings__c/fields/IsPlatformEventStorageEnabled__c.field-meta.xml @@ -1,12 +1,14 @@ IsPlatformEventStorageEnabled__c + DeprecateCandidate true Controls if LogEntryEvent__e platform events are transformed & stored in the custom objects Log__c and LogEntry__c (when IsSavingEnabled__c == true). false Controls if LogEntryEvent__e platform events are transformed & stored in the custom objects Log__c and LogEntry__c (when IsSavingEnabled__c == true). - + + false Checkbox diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls new file mode 100644 index 000000000..e2422ae32 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls @@ -0,0 +1,3 @@ +@SuppressWarnings('PMD.ApexDoc') +public class LoggerEmailUtils { +} diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls-meta.xml b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls new file mode 100644 index 000000000..71f82465d --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls @@ -0,0 +1,4 @@ +@SuppressWarnings('PMD.ApexDoc') +@IsTest(IsParallel=true) +private class LoggerEmailUtils_Tests { +} diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls-meta.xml b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerEmailUtils_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls new file mode 100644 index 000000000..86473e205 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls @@ -0,0 +1,3 @@ +@SuppressWarnings('PMD.ApexDoc') +public class LoggerSObjectHandlerPlugin { +} diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls-meta.xml b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls new file mode 100644 index 000000000..249852bb1 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls @@ -0,0 +1,4 @@ +@SuppressWarnings('PMD.ApexDoc') +@IsTest(IsParallel=true) +private class LoggerSObjectHandlerPlugin_Tests { +} diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls-meta.xml b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerSObjectHandlerPlugin_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls new file mode 100644 index 000000000..a0d8e3f6e --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls @@ -0,0 +1,4 @@ +@SuppressWarnings('PMD.ApexDoc') +@IsTest(IsParallel=true) +private class LoggerTestUtils { +} diff --git a/nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls-meta.xml b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/managed-package/core/main/deprecated/classes/LoggerTestUtils.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/managed-package/sfdx-project.json b/nebula-logger/managed-package/sfdx-project.json index 16229ea42..544fac143 100644 --- a/nebula-logger/managed-package/sfdx-project.json +++ b/nebula-logger/managed-package/sfdx-project.json @@ -10,9 +10,9 @@ "definitionFile": "./config/scratch-orgs/base-scratch-def.json", "postInstallScript": "LoggerInstallHandler", "ancestorVersion": "HIGHEST", - "versionNumber": "4.7.0.NEXT", - "versionName": "Spring '22 Release", - "versionDescription": "View the v4.7.0 milestone in GitHub for the list of changes - https://github.com/jongpie/NebulaLogger/milestone/7", + "versionNumber": "4.8.0.NEXT", + "versionName": "BETA - Summer '22 Release", + "versionDescription": "View the v4.8.0 milestone in GitHub for the list of changes - https://github.com/jongpie/NebulaLogger/milestone/7", "releaseNotesUrl": "https://github.com/jongpie/NebulaLogger/releases" } ], diff --git a/nebula-logger/plugins/.images/btn-install-unlocked-package-plugin-sandbox.png b/nebula-logger/plugins/.images/btn-install-unlocked-package-plugin-sandbox.png new file mode 100644 index 0000000000000000000000000000000000000000..6b271f988bca0b4cf9af9dac9a2f9896c2e28510 GIT binary patch literal 39945 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 literal 0 HcmV?d00001 diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/images/btn-install-unlocked-package-plugin.png b/nebula-logger/plugins/.images/btn-install-unlocked-package-plugin.png similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/images/btn-install-unlocked-package-plugin.png rename to nebula-logger/plugins/.images/btn-install-unlocked-package-plugin.png diff --git a/nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls b/nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls deleted file mode 100644 index aebf133ed..000000000 --- a/nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls +++ /dev/null @@ -1,186 +0,0 @@ -//------------------------------------------------------------------------------------------------// -// This file is part of the Nebula Logger project, released under the MIT License. // -// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // -//------------------------------------------------------------------------------------------------// - -// TODO: need to improve plugin framework to make it easier for plugins to test by making it easier -// to mock CMDT records & instances of LoggerSObjectHandlerPlugin -@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions, PMD.NcssMethodCount') -@IsTest -private class SlackLoggerPlugin_Tests { - /** - * @description Inner class for handling mock HTTP callouts. - */ - public class SuccessCalloutMock implements HttpCalloutMock { - /** - * @description Returns the mock HTTP response. - * @param request The HTTP request to "receive" - * @return The resulting HTTP response - */ - public HttpResponse respond(HttpRequest request) { - HttpResponse response = new HttpResponse(); - response.setBody(request.getBody()); - response.setStatusCode(200); - return response; - } - } - - /** - * @description Inner class for handling mock HTTP callouts that should end in errors / exceptions. - */ - public class FailureCalloutMock implements HttpCalloutMock { - /** - * @description Returns the mock HTTP response. - * @param request The HTTP request to "receive" - * @return The resulting HTTP response - */ - public HttpResponse respond(HttpRequest request) { - HttpResponse response = new HttpResponse(); - response.setBody(request.getBody()); - response.setStatusCode(400); - return response; - } - } - - static void mockConfigurations(LoggingLevel notificationLoggingLevel) { - // Set the plugin's parameters - LoggerPlugin__mdt slackPluginConfig = new LoggerPlugin__mdt( - IsEnabled__c = true, - PluginApiName__c = SlackLoggerPlugin.class.getName(), - PluginType__c = 'Apex' - ); - LoggerSObjectHandler.setMockPlugin(Schema.Log__c.SObjectType, slackPluginConfig); - - // Set the plugin's internal variables - SlackLoggerPlugin.endpoint = 'https://fake.slack.com/'; - SlackLoggerPlugin.notificationLoggingLevel = notificationLoggingLevel; - } - - static void verifyLogEntryCountEquals(Integer expectedCount) { - List existingLogEntries = [SELECT Id FROM LogEntry__c]; - System.assertEquals(expectedCount, existingLogEntries.size(), 'Existing log entries did NOT match the expected count.'); - } - - static List queryLogs() { - return [ - SELECT - Id, - MaxLogEntryLoggingLevelOrdinal__c, - SendSlackNotification__c, - SlackNotificationDate__c, - ( - SELECT Id, LoggingLevel__c, Message__c - FROM LogEntries__r - WHERE LoggingLevelOrdinal__c >= :LoggingLevel.WARN.ordinal() - ORDER BY Timestamp__c DESC - LIMIT 1 - ) - FROM Log__c - ]; - } - - @IsTest - static void pushLogWhenLoggingLevelIsMet() { - SuccessCalloutMock calloutMock = new SuccessCalloutMock(); - - verifyLogEntryCountEquals(0); - - Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), SendSlackNotification__c = false, TransactionId__c = '1234'); - insert log; - - LoggingLevel logEntryLoggingLevel = LoggingLevel.WARN; - LogEntry__c logEntry = new LogEntry__c( - Log__c = log.Id, - LoggingLevel__c = logEntryLoggingLevel.name(), - LoggingLevelOrdinal__c = logEntryLoggingLevel.ordinal(), - Timestamp__c = System.now() - ); - insert logEntry; - - verifyLogEntryCountEquals(1); - - List logs = queryLogs(); - System.assertEquals(1, logs.size(), 'Logs size did not match expected value of 1.'); - log = logs.get(0); - - System.assertEquals(1, log.LogEntries__r.size(), 'Log entries did not match the expected count of 1.'); - System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to true.'); - System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); - - Test.startTest(); - Test.setMock(HttpCalloutMock.class, calloutMock); - - // Load the mock configurations - the plugin framework won't load actual CMDT records during tests - mockConfigurations(logEntryLoggingLevel); - System.assert( - logEntryLoggingLevel.ordinal() >= SlackLoggerPlugin.notificationLoggingLevel.ordinal(), - 'The notification logging level ordinal was incorrect.' - ); - - // Update the records to trigger the handler framework, which will then run the Slack plugin - update log; - - // Verify that the internal queueable job has been enqueued - System.assertEquals(1, Limits.getAsyncCalls(), 'The enqueueable job was not enqueued.'); - - // Stop the test so the internal queueable job runs - Test.stopTest(); - - log = [SELECT Id, MaxLogEntryLoggingLevelOrdinal__c, SendSlackNotification__c, SlackNotificationDate__c FROM Log__c]; - System.assertEquals(true, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to false.'); - System.assertNotEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was null.'); - System.assertEquals(System.today(), log.SlackNotificationDate__c.date(), 'SlackNotificationDate was not set to TODAY.'); - } - - @IsTest - static void doNotPushLogWhenLoggingLevelIsNotMet() { - SuccessCalloutMock calloutMock = new SuccessCalloutMock(); - - verifyLogEntryCountEquals(0); - - Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), SendSlackNotification__c = false, TransactionId__c = '1234'); - insert log; - - LoggingLevel logEntryLoggingLevel = LoggingLevel.WARN; - LogEntry__c logEntry = new LogEntry__c( - Log__c = log.Id, - LoggingLevel__c = logEntryLoggingLevel.name(), - LoggingLevelOrdinal__c = logEntryLoggingLevel.ordinal(), - Timestamp__c = System.now() - ); - insert logEntry; - - verifyLogEntryCountEquals(1); - - List logs = queryLogs(); - System.assertEquals(1, logs.size(), 'Logs size did not match expected value of 1.'); - log = logs.get(0); - - System.assertEquals(1, log.LogEntries__r.size(), 'Log entries size was not equal to 1.'); - System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to true.'); - System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); - - Test.startTest(); - Test.setMock(HttpCalloutMock.class, calloutMock); - - // Load the mock configurations - the plugin framework won't load actual CMDT records during tests - LoggingLevel slackLoggingLevel = LoggingLevel.ERROR; - System.assert(logEntryLoggingLevel.ordinal() < slackLoggingLevel.ordinal(), 'Slack logging level ordinal was incorrect.'); - mockConfigurations(slackLoggingLevel); - System.assert(logEntryLoggingLevel.ordinal() < SlackLoggerPlugin.notificationLoggingLevel.ordinal(), 'Slack logging level ordinal was incorrect.'); - - // Update the records to trigger the handler framework, which will then run the Slack plugin - update log; - - // Verify that the internal queueable job has been enqueued - System.assertEquals(0, Limits.getAsyncCalls(), 'The queueable job has not been enqueued.'); - - // Stop the test so the internal queueable job runs - Test.stopTest(); - - log = queryLogs().get(0); - System.assertEquals(1, log.LogEntries__r.size(), 'Log entries size was not equal to 1.'); - System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification incorrectly set to true.'); - System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); - } -} diff --git a/nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml b/nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml deleted file mode 100644 index 7160cebf7..000000000 --- a/nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml +++ /dev/null @@ -1,27 +0,0 @@ - - - - false - - Description__c - Adds a Slack integration for Nebula Logger. - -Any logs with MaxLogEntryLoggingLevelOrdinal__c >= the parameter 'SlackLoggingLevelThreshold' will send a notification to Slack - - - PluginApiName__c - SlackLoggerPlugin - - - PluginType__c - Apex - - - SObjectType__c - Log__c - - diff --git a/nebula-logger/plugins/big-object-archiving/.images/log-entry-archives-tab.png b/nebula-logger/plugins/big-object-archiving/.images/log-entry-archives-tab.png new file mode 100644 index 0000000000000000000000000000000000000000..b1649e86fcd9d59c6e2c3c79776c84c5f81645d3 GIT binary patch literal 226207 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

DafqvMs^JBtc(o=!qU z+byc28kGRc&h)vOr=RA;JvBY~ZrXrU{mz7bPyIQGTH0wk8s<{aU-|M!3^liVy9N-F ztvcmxrF?v-b0M~E4Q(7rtH0X3ac%bZG39M!;wnzJh(j+?A)iTTd2?HRO_@@?+6@R4 zb)uQEOlQ*sexpynXnpS*omyA*SnP42;74%`mE7U|zPdfajLbgiOgO(eDggAEoV^hC zmX`v)>%_TuJlL;`X;Q6ySTT$tsr2DWIFY7KihQTOfs|cFdYRP!R8lS&P~%u`dVhU^ z_PZ@TChF2P79VRFFMs*@bkP%7Xb+u=D(a;&4&9mAa^R}~60Ul(qd*1{uHVzP zA3TOU{Os@F`7J}6wsAvSI#0Lv?2?}u)V8&I-GG}pXi{4kTt;7`j%@BEB4puQ_wuI-b4)+-O{ zf`t5(j-ZgE9w$P*8IsL1I_gBmckz-l!&Y5YZAjhIWoFTHgf!|Cj!m-Ob_|p!U?$mtER!rPZ_#(99Y41&H$GoUIWJC=E%bSMbzYiG=s6 zXKm#~@y4~IB)_tO64ZE5p1WqLa2ts#U~ChX*`NPeBY6q9t8w`uiN2UiI!{lc`TO7+ zO=Uh$0hT}YkM=8V=*pEJAO!40PnH~%nK%HHemH1{*@V(#GL$63anjvnaUdkO9Y^cK474Q zz#s=h3uLg=E|(?%KvLgm8+TNqtBB^J3I6n?3tckrokX67(R`f3g@8b|qxZ@E7n)C- z_0ffb!#d#i0n#EcueXwOk0su?Wchr?OW3Zwh(KT8WnSr!Xd@PHK$%1r?JzHeZ~TNZ z>U%}N1^;GY@gqL|h0$7W4~8^7vA%Hp-WTsdg5ec?H|3g8fp1*W0}_AQol6cm)xIn$W6%UMHJ{^!a*QXo^UZ8FNi6i*|kD#53Zzynx-s||o! zpce{jf{&q1KgLT6oIW$%=Zb2(5^~q*6E=i~I+s3Y&SvyJzR}j(5|3%-!BNUMpa3K% z>@-e?%oN6~!Tfr2Q8gDO5yV7CSlRFsu6U2fL{^_<4`UY9InEd&9y3Si?(uM2VU}NH zd&aY+%e=Gq3kF5@M};F`T3-@70+RnapdC?9PmjmLXHMTP4av1Hr?vL%b+1i_wAo32 zv-Ud#vnoq=cHf@z+Bqv^$*emd2E9WA90`?7L`yi9&BlvI5$3O;=sRdBVw&m{H~R-o z1yJ6{8rRJvl9w#QrJv?dg=zI6l4^0iP_<_055M!DRYt>Ge?@PP&Pl4NOx3UdfM6HNpcxi*c^OLOwC_l6 zXS*0mbMRv^B)4$~zf*Zm-QDS10BVg_G-|$$>f8PKwMAGL*SXj>v&~1rPbEfwOYp`= zIvix7yPZ6sIw2<{B-CK3{&efaWZ3#i!yE{Ot1HisD{JSsASrOn@(0M%cS#_jGIcN} z4^^5C#K|9?SH7ss8}PO@7XiTSDuIi+xVYNa9J;Yi0DSoP7@MPee}500PQ} z6B|G^gKco#g)Z6+R8z)0>9o^#+=-o?5_(SuZ!a#d#3*qcZYM2w_FKN_#bffzFn?CV zXmCS^K0oeUdDzRUIIj7qbHCVia}t%6r<2#I(T1$AySK`-*|`DR*$GM@%kaM@+soM_ z@ZYz$8DClm@9I+#TbK)7BF5GWBK>xFi_?%Kgdh1`o|_}ZDXz}vWJdYSuJvsM|3TRc zkxh61rSV)@iAE;Q2H_wv)@%bu=%gJ6AhY@04)8%n^Ht^ml0^8f^WpBDrlxqQR$fsN zy|FDZ=<9$PJ~44OpjgLih=Mo@*e&f##cqOWGO zCT};N?%XBxH_LeiradAb3`+tP%tqk%A_#@l!Q-0o?!R9 z&OthrU;a^FmcFr3wKx&{sr#(3@C_t1Mp!{3pZhM_J-yS?)}2ou>Mx7k(}Ul zTZS;<7K0dBG9MiKwnrY&>VzzrvTmFm)0>+9sM^&>fEp8Ec8rw$gtFXp)W~Rxv|P91 zD>$Ri2|8G5BLm`fxs@B8gbulT)8fD;dZG`UOW zsI3BFPH`v(7GT3pwy#%0TB>0<8erG5(F5^i#Koa3M$&?xL}<${uf;y@^t}mo@3<`} zUu|iTJ$14|OEN^>$Zf#(S`<%t+9DM4VX0Rk4XQuJQNkjBaMxWe)eAR0et7L`h!Hd@ z?s11efhfj@-?_Rq-Az}6Abi>NV2FCw_)(fvp zTHS|a&t$}bS%=7G1RNISYNKvQHru&?wL92wwn*eyRYvN1EHhJ7LP%|D^FFvypgYmD zW1(>%|0uoUP3zcNjtYA$I=lmk={&hrj?Jb1vKF2{&|E_!kP_!~LjQXC$mJ*dFN?HhC$lKZzUr5g*hZ$hPQu zxaXdXS%+nb{CRz0R(Uakyb$5ve7YV9ayysN^E^RGS%k+CX@n~1DY&^OYYSaowi83X zxE56Rc}7l1OqK+gC_Xz1VziLLsRa`W&_Ba zkw1l^kJf|iRslq%TzRL2b_?}KdQIVGTJVEzN zi@^LXvq|r*LS^{hE=ca;xLJG?r#M&T;>^rzn>9u3*^3!8F3peKCvF=rxesVc1nppm z1IUbt`|TKJbociLz)02l}?M7s{|G+D=YfS z-AvudviETgKv~UnCQpv-y5s^q4Y5CsHt=!S{eYlex*EVG@fCU>KYlDx++1H@|K$=R z6t~7a5HO(|UAX9T4GA%P+|&L+$0%M0K}%JF7RMluA6Pwdrx?=!mD7TP!Eovf;y|5V zCWw$d^C5d8jUqW5F+z;)AY^DdW`DL!oJLcucv=Dx!SPZ7LBwyYWykqSqDX5kukBl> z;q#@Ts=zW(Rch-d-hhh;pC<(PrgN)4JTh_ftmH^5pN=eg1^s0{ke0(Lh2DkSg;bHi z{l()G7nM3pZ$^uYiU`=l3f1UL!rGOHn0rs-6bcYwq4QWcKb6c+3^tMsr*hI-gK6R$ zySBErascu=hx**5mXJ49NvFYydX$U1!)!1ZTpHNN87fBZA6F&aKQ`7^vn-O3oRnAY z2eNQxS+Lcfn4b!6!h261?k~D3zH(_mEbN7U{ztRjj14*mH2O^>Xl6$7y!_0bD}8 z*X4(U=iOT17l4$ALD{qWg)L9PD0n77TQe)uruVeP8!*)DcY~)F&p(1Ba>T8z(H=TL zT!gsfP)R;^^xt4$Ffd^|jqBmR0c#;M9RCz^blh~F0@!1t@t9zg*V%=AXp1$Z31g`d zz_KdzGry=jYdA2=A@LEk=-2sq^}jd=q%S_1zFt5Gs$WMIW5iYakwmFYVIj!%kEhMG zb_SZe>j(4IpsySg>B@c4B$75ZjIWt=*tV*G*6PH}43CA)a;02>#yf3oRsPIV4LY2u z1uwVyw6wIjl;-*HH$(#v51^ll4r<@HT>mg6t#^DFJhc_j@ioYSFEk*)7vN3uthCsf z;H1AQN+YhZ>6Q6%5sQW=DQ2y$g^+zwpqFLC|7=hka1|{`9;@h(^T?rKITkj}jw=DM zld25~6j8Gea}MJH&OT2rxT&wn%9z_9lLpX1$}e8F>wK{@Vco(HR7vTm!NI}vXn=X{ z)OP(1!$9xB9c_CoD`fHT3&c)Uw_)PEa>OG2r15nG977`V`Ja)lhuiZhtQd%0AHcKI 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π}|eOW>{d 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

    U8h`$6#%OZk5BLJqF86EEmV}6a9Mktc3M^m3VpJ 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 JvJRaI{vTt{h?@Wa literal 0 HcmV?d00001 diff --git a/nebula-logger/plugins/big-object-archiving/README.md b/nebula-logger/plugins/big-object-archiving/README.md new file mode 100644 index 000000000..403644da2 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/README.md @@ -0,0 +1,32 @@ +# Big Object Archiving plugin for Nebula Logger + +> :information_source: This plugin requires `v4.7.1` or newer of Nebula Logger's unlocked package + +[![Install Unlocked Package](../.images/btn-install-unlocked-package-plugin-sandbox.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015lgLQAQ) +Adds a Big Object - `LogEntryArchive__b` - to Nebula Logger for archiving of logging data. + +The `LogEntryArchive__b` object can be used to: + +- **To immediately publish logging data, instead of using platform events**. Out of the box, Nebula Logger leverages platform events (`LogEntryEvent__e`) as the mechanism for saving logging data, even if an error occurs. It then transforms the platform event data in `LogEntryEvent__e` and stores it in `Log__c` and `LogEntry__c` for reporting & managing of logs. If you want to skip the platform events altogether, and don't need to report on the data via Log**c or LogEntry**c, then you can leverage the Big Object Archiving plugin to immediately write to `LogEntryArchive__b`, giving you a way to store logging data on platform, without consuming your data storage limits (although it's worth noting that Big Objects have their own storage limits as well). +- **To later archive `Log__c` and `LogEntry__c` data**. If you want to leverage the benefits of having logging data in `Log__c` and `LogEntry__c`, but need to delete some records due to storage limits, the plugin also provides the ability to archive `Log__c` and `LogEntry__c` data into the big object `LogEntryArchive__b`. This is accomplished by setting a "Log Purge Action" to "Archive" on any `Log__c` records that you want to archive into the `LogEntryArchive__b` big object. + - When the `LogBatchPurger` job runs, any `Log__c` (and related `LogEntry__c` records) with a "Log Purge Action" of "Archive" and a "Log Retention Date" <= TODAY will first be copied into the big object `LogEntryArchive__b`, before being deleted in `Log__c` and `LogEntry__c`. + +![Big Object Archiving plugin: Log Entry Archives tab](./images/log-entry-archives-tab.png) + +--- + +## What's Included + +This plugin includes some add-on metadata for Nebula Logger to support storing logging data within a big object + +1. BigObject `LogEntryArchive__b` +2. Custom tab "Log Entry Archives" to display the included LWC `logEntryArchives` +3. Apex classes `LogEntryArchivePlugin`, `LogEntryArchiveBuilder` and corresponding test classes +4. Plugin configuration details stored in Logger's CMDT objects `LoggerPlugin__mdt` and `LoggerParameter__mdt` +5. Field-level security (FLS) & tab access via a new permission set `LogEntryArchiveAdmin` to provide access to the custom Slack fields + +--- + +## Installation Steps + +TODO diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls new file mode 100644 index 000000000..0c841fe60 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls @@ -0,0 +1,325 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Plugins + * @description Builder class to create an instance of `LogEntryArchive__b`, used by the BigObject plugin + * @see LogEntryArchivePlugin + * @see LogEntryEventBuilder + */ +public without sharing class LogEntryArchiveBuilder { + private LogEntryEvent__e logEntryEvent; + private LogEntry__c logEntry; + private LogEntryArchive__b logEntryArchive; + + /** + * @description Used by `LogEntryArchivePlugin` to instantiate a new instance of `LogEntryArchiveBuilder` + * @param logEntryEvent The `LogEntryEvent__e` record that will be converted to a `LogEntryArchive__b` record + */ + public LogEntryArchiveBuilder(LogEntryEvent__e logEntryEvent) { + this.logEntryEvent = logEntryEvent; + } + + /** + * @description Used by `LogEntryArchivePlugin` to instantiate a new instance of `LogEntryArchiveBuilder` + * @param logEntry The `LogEntry__c` record that will be converted to a `LogEntryArchive__b` record + */ + public LogEntryArchiveBuilder(LogEntry__c logEntry) { + this.logEntry = logEntry; + } + + /** + * @description Returns the `LogEntryArchive__b` record for this instance of LogEntryEventBuilder + * @return The `LogEntryArchive__b` record + */ + public LogEntryArchive__b getLogEntryArchive() { + if (this.logEntryArchive == null && this.logEntryEvent != null) { + this.convertLogEntryEventToLogEntryArchive(); + this.truncateFieldValues(); + } else if (this.logEntryArchive == null && this.logEntry != null) { + this.convertLogEntryToLogEntryArchive(); + this.truncateFieldValues(); + } + + return this.logEntryArchive; + } + + private void convertLogEntryEventToLogEntryArchive() { + this.logEntryArchive = new LogEntryArchive__b( + ApiVersion__c = this.logEntryEvent.ApiVersion__c, + ArchivedById__c = UserInfo.getUserId(), + ArchivedDate__c = System.now(), + ArchivedByUsername__c = UserInfo.getUsername(), + ComponentType__c = this.logEntryEvent.ComponentType__c, + DatabaseResultCollectionSize__c = this.logEntryEvent.DatabaseResultCollectionSize__c, + DatabaseResultCollectionType__c = this.logEntryEvent.DatabaseResultCollectionType__c, + DatabaseResultJson__c = this.logEntryEvent.DatabaseResultJson__c, + DatabaseResultType__c = this.logEntryEvent.DatabaseResultType__c, + EpochTimestamp__c = this.logEntryEvent.EpochTimestamp__c, + EventUuid__c = this.logEntryEvent.EventUuid, + ExceptionMessage__c = this.logEntryEvent.ExceptionMessage__c, + ExceptionStackTrace__c = this.logEntryEvent.ExceptionStackTrace__c, + ExceptionType__c = this.logEntryEvent.ExceptionType__c, + HttpRequestBody__c = this.logEntryEvent.HttpRequestBody__c, + HttpRequestBodyMasked__c = String.valueOf(this.logEntryEvent.HttpRequestBodyMasked__c), + HttpRequestCompressed__c = String.valueOf(this.logEntryEvent.HttpRequestCompressed__c), + HttpRequestEndpoint__c = this.logEntryEvent.HttpRequestEndpoint__c, + HttpRequestMethod__c = this.logEntryEvent.HttpRequestMethod__c, + HttpResponseBody__c = this.logEntryEvent.HttpResponseBody__c, + HttpResponseBodyMasked__c = String.valueOf(this.logEntryEvent.HttpResponseBody__c), + HttpResponseHeaderKeys__c = this.logEntryEvent.HttpResponseHeaderKeys__c, + HttpResponseStatus__c = this.logEntryEvent.HttpResponseStatus__c, + HttpResponseStatusCode__c = this.logEntryEvent.HttpResponseStatusCode__c, + LimitsAggregateQueriesMax__c = this.logEntryEvent.LimitsAggregateQueriesMax__c, + LimitsAggregateQueriesUsed__c = this.logEntryEvent.LimitsAggregateQueriesUsed__c, + LimitsAsyncCallsMax__c = this.logEntryEvent.LimitsAsyncCallsMax__c, + LimitsAsyncCallsUsed__c = this.logEntryEvent.LimitsAsyncCallsUsed__c, + LimitsCalloutsUsed__c = this.logEntryEvent.LimitsCalloutsUsed__c, + LimitsCpuTimeMax__c = this.logEntryEvent.LimitsCpuTimeMax__c, + LimitsCpuTimeUsed__c = this.logEntryEvent.LimitsCpuTimeUsed__c, + LimitsDmlRowsMax__c = this.logEntryEvent.LimitsDmlRowsMax__c, + LimitsDmlRowsUsed__c = this.logEntryEvent.LimitsDmlRowsUsed__c, + LimitsDmlStatementsMax__c = this.logEntryEvent.LimitsDmlStatementsMax__c, + LimitsDmlStatementsUsed__c = this.logEntryEvent.LimitsDmlStatementsUsed__c, + LimitsEmailInvocationsMax__c = this.logEntryEvent.LimitsEmailInvocationsMax__c, + LimitsEmailInvocationsUsed__c = this.logEntryEvent.LimitsEmailInvocationsUsed__c, + LimitsFutureCallsMax__c = this.logEntryEvent.LimitsFutureCallsMax__c, + LimitsFutureCallsUsed__c = this.logEntryEvent.LimitsFutureCallsUsed__c, + LimitsHeapSizeMax__c = this.logEntryEvent.LimitsHeapSizeMax__c, + LimitsHeapSizeUsed__c = this.logEntryEvent.LimitsHeapSizeUsed__c, + LimitsMobilePushApexCallsMax__c = this.logEntryEvent.LimitsMobilePushApexCallsMax__c, + LimitsMobilePushApexCallsUsed__c = this.logEntryEvent.LimitsMobilePushApexCallsUsed__c, + LimitsPublishImmediateDmlStatementsMax__c = this.logEntryEvent.LimitsPublishImmediateDmlStatementsMax__c, + LimitsPublishImmediateDmlStatementsUsed__c = this.logEntryEvent.LimitsPublishImmediateDmlStatementsUsed__c, + LimitsQueueableJobsMax__c = this.logEntryEvent.LimitsQueueableJobsMax__c, + LimitsQueueableJobsUsed__c = this.logEntryEvent.LimitsQueueableJobsUsed__c, + LimitsSoqlQueriesMax__c = this.logEntryEvent.LimitsSoqlQueriesMax__c, + LimitsSoqlQueriesUsed__c = this.logEntryEvent.LimitsSoqlQueriesUsed__c, + LimitsSoqlQueryLocatorRowsMax__c = this.logEntryEvent.LimitsSoqlQueryLocatorRowsMax__c, + LimitsSoqlQueryLocatorRowsUsed__c = this.logEntryEvent.LimitsSoqlQueryLocatorRowsUsed__c, + LimitsSoqlQueryRowsMax__c = this.logEntryEvent.LimitsSoqlQueryRowsMax__c, + LimitsSoqlQueryRowsUsed__c = this.logEntryEvent.LimitsSoqlQueryRowsUsed__c, + LimitsSoslSearchesMax__c = this.logEntryEvent.LimitsSoslSearchesMax__c, + LimitsSoslSearchesUsed__c = this.logEntryEvent.LimitsSoslSearchesUsed__c, + Locale__c = this.logEntryEvent.Locale__c, + // LoggedBy__c = String.isNotBlank(logEntryEvent.LoggedById__c) ? logEntryEvent.LoggedById__c : 'Anonymous', + LoggedById__c = this.logEntryEvent.LoggedById__c, + LoggedByUsername__c = String.isNotBlank(this.logEntryEvent.LoggedByUsername__c) ? this.logEntryEvent.LoggedByUsername__c : 'Anonymous', + LoggerVersionNumber__c = this.logEntryEvent.LoggerVersionNumber__c, + LoggingLevel__c = this.logEntryEvent.LoggingLevel__c, + LoggingLevelOrdinal__c = this.logEntryEvent.LoggingLevelOrdinal__c, + LoginApplication__c = this.logEntryEvent.LoginApplication__c, + LoginBrowser__c = this.logEntryEvent.LoginBrowser__c, + LoginHistoryId__c = this.logEntryEvent.LoginHistoryId__c, + LoginPlatform__c = this.logEntryEvent.LoginPlatform__c, + LoginType__c = this.logEntryEvent.LoginType__c, + LogoutUrl__c = this.logEntryEvent.LogoutUrl__c, + Message__c = this.logEntryEvent.Message__c, + NetworkId__c = this.logEntryEvent.NetworkId__c, + NetworkLoginUrl__c = this.logEntryEvent.NetworkLoginUrl__c, + NetworkLogoutUrl__c = this.logEntryEvent.NetworkLogoutUrl__c, + NetworkSelfRegistrationUrl__c = this.logEntryEvent.NetworkSelfRegistrationUrl__c, + NetworkUrlPathPrefix__c = this.logEntryEvent.NetworkUrlPathPrefix__c, + OrganizationDomainUrl__c = this.logEntryEvent.OrganizationDomainUrl__c, + OrganizationEnvironmentType__c = this.logEntryEvent.OrganizationEnvironmentType__c, + OrganizationId__c = this.logEntryEvent.OrganizationId__c, + OrganizationInstanceName__c = this.logEntryEvent.OrganizationInstanceName__c, + OrganizationName__c = this.logEntryEvent.OrganizationName__c, + OrganizationNamespacePrefix__c = this.logEntryEvent.OrganizationNamespacePrefix__c, + OrganizationType__c = this.logEntryEvent.OrganizationType__c, + OriginLocation__c = this.logEntryEvent.OriginLocation__c, + OriginType__c = this.logEntryEvent.OriginType__c, + ParentLogTransactionId__c = this.logEntryEvent.ParentLogTransactionId__c, + ProfileId__c = this.logEntryEvent.ProfileId__c, + ProfileName__c = this.logEntryEvent.ProfileName__c, + RecordCollectionSize__c = this.logEntryEvent.RecordCollectionSize__c, + RecordCollectionType__c = this.logEntryEvent.RecordCollectionType__c, + RecordId__c = this.logEntryEvent.RecordId__c, + RecordJson__c = this.logEntryEvent.RecordJson__c, + RecordSObjectClassification__c = this.logEntryEvent.RecordSObjectClassification__c, + RecordSObjectType__c = this.logEntryEvent.RecordSObjectType__c, + RecordSObjectTypeNamespace__c = this.logEntryEvent.RecordSObjectTypeNamespace__c, + SessionId__c = this.logEntryEvent.SessionId__c, + SessionSecurityLevel__c = this.logEntryEvent.SessionSecurityLevel__c, + SessionType__c = this.logEntryEvent.SessionType__c, + SourceIp__c = this.logEntryEvent.SourceIp__c, + StackTrace__c = this.logEntryEvent.StackTrace__c, + SystemMode__c = this.logEntryEvent.SystemMode__c, + Tags__c = this.logEntryEvent.Tags__c, + ThemeDisplayed__c = this.logEntryEvent.ThemeDisplayed__c, + Timestamp__c = this.logEntryEvent.Timestamp__c, + TimestampString__c = this.logEntryEvent.TimestampString__c, + TimeZoneId__c = this.logEntryEvent.TimeZoneId__c, + TransactionEntryNumber__c = this.logEntryEvent.TransactionEntryNumber__c, + TransactionId__c = this.logEntryEvent.TransactionId__c, + TriggerOperationType__c = this.logEntryEvent.TriggerOperationType__c, + TriggerSObjectType__c = this.logEntryEvent.TriggerSObjectType__c, + UserLicenseDefinitionKey__c = this.logEntryEvent.UserLicenseDefinitionKey__c, + UserLicenseName__c = this.logEntryEvent.UserLicenseName__c, + UserLoggingLevel__c = this.logEntryEvent.UserLoggingLevel__c, + UserLoggingLevelOrdinal__c = this.logEntryEvent.UserLoggingLevelOrdinal__c, + UserRoleId__c = this.logEntryEvent.UserRoleId__c, + UserRoleName__c = this.logEntryEvent.UserRoleName__c, + UserType__c = this.logEntryEvent.UserType__c + ); + } + + private void convertLogEntryToLogEntryArchive() { + List tagNames = new List(); + for (LogEntryTag__c logEntryTag : this.logEntry.LogEntryTags__r) { + tagNames.add(logEntryTag.Tag__r.Name); + } + tagNames = new List(new Set(tagNames)); + tagNames.sort(); + + this.logEntryArchive = new LogEntryArchive__b( + ApiReleaseNumber__c = this.logEntry.Log__r.ApiReleaseNumber__c, + ApiReleaseVersion__c = this.logEntry.Log__r.ApiReleaseVersion__c, + ApiVersion__c = this.logEntry.Log__r.ApiVersion__c, + ArchivedById__c = UserInfo.getUserId(), + ArchivedDate__c = System.now(), + ArchivedByUsername__c = UserInfo.getUsername(), + ClosedById__c = this.logEntry.Log__r.ClosedBy__c, + ClosedByUsername__c = this.logEntry.Log__r.ClosedBy__r.Username, + ClosedDate__c = this.logEntry.Log__r.ClosedDate__c, + Comments__c = this.logEntry.Log__r.Comments__c, + ComponentType__c = this.logEntry.ComponentType__c, + DatabaseResultCollectionSize__c = this.logEntry.DatabaseResultCollectionSize__c, + DatabaseResultCollectionType__c = this.logEntry.DatabaseResultCollectionType__c, + DatabaseResultJson__c = this.logEntry.DatabaseResultJson__c, + DatabaseResultType__c = this.logEntry.DatabaseResultType__c, + EpochTimestamp__c = this.logEntry.EpochTimestamp__c, + EventUuid__c = this.logEntry.EventUuid__c, + ExceptionMessage__c = this.logEntry.ExceptionMessage__c, + ExceptionStackTrace__c = this.logEntry.ExceptionStackTrace__c, + ExceptionType__c = this.logEntry.ExceptionType__c, + HttpRequestBody__c = this.logEntry.HttpRequestBody__c, + HttpRequestBodyMasked__c = String.valueOf(this.logEntry.HttpRequestBodyMasked__c), + HttpRequestCompressed__c = String.valueOf(this.logEntry.HttpRequestCompressed__c), + HttpRequestEndpoint__c = this.logEntry.HttpRequestEndpoint__c, + HttpRequestMethod__c = this.logEntry.HttpRequestMethod__c, + HttpResponseBody__c = this.logEntry.HttpResponseBody__c, + HttpResponseBodyMasked__c = String.valueOf(this.logEntry.HttpResponseBodyMasked__c), + HttpResponseHeaderKeys__c = this.logEntry.HttpResponseHeaderKeys__c, + HttpResponseStatus__c = this.logEntry.HttpResponseStatus__c, + HttpResponseStatusCode__c = this.logEntry.HttpResponseStatusCode__c, + IsClosed__c = String.valueOf(this.logEntry.Log__r.IsClosed__c), + IsResolved__c = String.valueOf(this.logEntry.Log__r.IsResolved__c), + Issue__c = this.logEntry.Log__r.Issue__c, + LimitsAggregateQueriesMax__c = this.logEntry.LimitsAggregateQueriesMax__c, + LimitsAggregateQueriesUsed__c = this.logEntry.LimitsAggregateQueriesUsed__c, + LimitsAsyncCallsMax__c = this.logEntry.LimitsAsyncCallsMax__c, + LimitsAsyncCallsUsed__c = this.logEntry.LimitsAsyncCallsUsed__c, + LimitsCalloutsUsed__c = this.logEntry.LimitsCalloutsUsed__c, + LimitsCpuTimeMax__c = this.logEntry.LimitsCpuTimeMax__c, + LimitsCpuTimeUsed__c = this.logEntry.LimitsCpuTimeUsed__c, + LimitsDmlRowsMax__c = this.logEntry.LimitsDmlRowsMax__c, + LimitsDmlRowsUsed__c = this.logEntry.LimitsDmlRowsUsed__c, + LimitsDmlStatementsMax__c = this.logEntry.LimitsDmlStatementsMax__c, + LimitsDmlStatementsUsed__c = this.logEntry.LimitsDmlStatementsUsed__c, + LimitsEmailInvocationsMax__c = this.logEntry.LimitsEmailInvocationsMax__c, + LimitsEmailInvocationsUsed__c = this.logEntry.LimitsEmailInvocationsUsed__c, + LimitsFutureCallsMax__c = this.logEntry.LimitsFutureCallsMax__c, + LimitsFutureCallsUsed__c = this.logEntry.LimitsFutureCallsUsed__c, + LimitsHeapSizeMax__c = this.logEntry.LimitsHeapSizeMax__c, + LimitsHeapSizeUsed__c = this.logEntry.LimitsHeapSizeUsed__c, + LimitsMobilePushApexCallsMax__c = this.logEntry.LimitsMobilePushApexCallsMax__c, + LimitsMobilePushApexCallsUsed__c = this.logEntry.LimitsMobilePushApexCallsUsed__c, + LimitsPublishImmediateDmlStatementsMax__c = this.logEntry.LimitsPublishImmediateDmlStatementsMax__c, + LimitsPublishImmediateDmlStatementsUsed__c = this.logEntry.LimitsPublishImmediateDmlStatementsUsed__c, + LimitsQueueableJobsMax__c = this.logEntry.LimitsQueueableJobsMax__c, + LimitsQueueableJobsUsed__c = this.logEntry.LimitsQueueableJobsUsed__c, + LimitsSoqlQueriesMax__c = this.logEntry.LimitsSoqlQueriesMax__c, + LimitsSoqlQueriesUsed__c = this.logEntry.LimitsSoqlQueriesUsed__c, + LimitsSoqlQueryLocatorRowsMax__c = this.logEntry.LimitsSoqlQueryLocatorRowsMax__c, + LimitsSoqlQueryLocatorRowsUsed__c = this.logEntry.LimitsSoqlQueryLocatorRowsUsed__c, + LimitsSoqlQueryRowsMax__c = this.logEntry.LimitsSoqlQueryRowsMax__c, + LimitsSoqlQueryRowsUsed__c = this.logEntry.LimitsSoqlQueryRowsUsed__c, + LimitsSoslSearchesMax__c = this.logEntry.LimitsSoslSearchesMax__c, + LimitsSoslSearchesUsed__c = this.logEntry.LimitsSoslSearchesUsed__c, + Locale__c = this.logEntry.Log__r.Locale__c, + LogEntryName__c = this.logEntry.Name, + LoggedById__c = this.logEntry.Log__r.LoggedBy__c, + LoggedByUsername__c = this.logEntry.Log__r.LoggedByUsername__c, + LoggerVersionNumber__c = this.logEntry.Log__r.LoggerVersionNumber__c, + LoggingLevel__c = this.logEntry.LoggingLevel__c, + LoggingLevelOrdinal__c = this.logEntry.LoggingLevelOrdinal__c, + LoginApplication__c = this.logEntry.Log__r.LoginApplication__c, + LoginBrowser__c = this.logEntry.Log__r.LoginBrowser__c, + LoginHistoryId__c = this.logEntry.Log__r.LoginHistoryId__c, + LoginPlatform__c = this.logEntry.Log__r.LoginPlatform__c, + LoginType__c = this.logEntry.Log__r.LoginType__c, + LogName__c = this.logEntry.Log__r.Name, + LogoutUrl__c = this.logEntry.Log__r.LogoutUrl__c, + LogPurgeAction__c = this.logEntry.Log__r.LogPurgeAction__c, // TODO Not sure it makes sense to have these fields in archive? Maybe? + LogRetentionDate__c = this.logEntry.Log__r.LogRetentionDate__c, // TODO Not sure it makes sense to have these fields in archive? Maybe? + Message__c = this.logEntry.Message__c, + MessageMasked__c = String.valueOf(this.logEntry.MessageMasked__c), + MessageTruncated__c = String.valueOf(this.logEntry.MessageTruncated__c), + NetworkId__c = this.logEntry.Log__r.NetworkId__c, + NetworkLoginUrl__c = this.logEntry.Log__r.NetworkLoginUrl__c, + NetworkLogoutUrl__c = this.logEntry.Log__r.NetworkLogoutUrl__c, + NetworkSelfRegistrationUrl__c = this.logEntry.Log__r.NetworkSelfRegistrationUrl__c, + NetworkUrlPathPrefix__c = this.logEntry.Log__r.NetworkUrlPathPrefix__c, + OrganizationDomainUrl__c = this.logEntry.Log__r.OrganizationDomainUrl__c, + OrganizationEnvironmentType__c = this.logEntry.Log__r.OrganizationEnvironmentType__c, + OrganizationId__c = this.logEntry.Log__r.OrganizationId__c, + OrganizationInstanceName__c = this.logEntry.Log__r.OrganizationInstanceName__c, + OrganizationInstanceReleaseCycle__c = this.logEntry.Log__r.OrganizationInstanceReleaseCycle__c, // TODO deprecated field + OrganizationName__c = this.logEntry.Log__r.OrganizationName__c, + OrganizationNamespacePrefix__c = this.logEntry.Log__r.OrganizationNamespacePrefix__c, + OrganizationType__c = this.logEntry.Log__r.OrganizationType__c, + OriginLocation__c = this.logEntry.OriginLocation__c, + OriginType__c = this.logEntry.OriginType__c, + ParentLogTransactionId__c = this.logEntry.Log__r.ParentLog__r?.TransactionId__c, + Priority__c = this.logEntry.Log__r.Priority__c, + ProfileId__c = this.logEntry.Log__r.ProfileId__c, + ProfileName__c = this.logEntry.Log__r.ProfileName__c, + RecordCollectionSize__c = this.logEntry.RecordCollectionSize__c, + RecordCollectionType__c = this.logEntry.RecordCollectionType__c, + RecordId__c = this.logEntry.RecordId__c, + RecordJson__c = this.logEntry.RecordJson__c, + RecordJsonMasked__c = String.valueOf(this.logEntry.RecordJsonMasked__c), + RecordName__c = String.valueOf(this.logEntry.RecordName__c), + RecordSObjectClassification__c = this.logEntry.RecordSObjectClassification__c, + RecordSObjectType__c = this.logEntry.RecordSObjectType__c, + RecordSObjectTypeNamespace__c = this.logEntry.RecordSObjectTypeNamespace__c, + SessionId__c = this.logEntry.Log__r.SessionId__c, + SessionSecurityLevel__c = this.logEntry.Log__r.SessionSecurityLevel__c, + SessionType__c = this.logEntry.Log__r.SessionType__c, + SourceIp__c = this.logEntry.Log__r.SourceIp__c, + StackTrace__c = this.logEntry.StackTrace__c, + Status__c = this.logEntry.Log__r.Status__c, + SystemMode__c = this.logEntry.Log__r.SystemMode__c, + Tags__c = String.join(tagNames, '\n'), + ThemeDisplayed__c = this.logEntry.Log__r.ThemeDisplayed__c, + Timestamp__c = this.logEntry.Timestamp__c, + TimestampString__c = String.valueOf(this.logEntry.Timestamp__c), + TimeZoneId__c = this.logEntry.Log__r.TimeZoneId__c, + TransactionEntryNumber__c = this.logEntry.TransactionEntryNumber__c, + TransactionId__c = this.logEntry.Log__r.TransactionId__c, + TriggerIsExecuting__c = String.valueOf(this.logEntry.TriggerIsExecuting__c), + TriggerOperationType__c = this.logEntry.TriggerOperationType__c, + TriggerSObjectType__c = this.logEntry.TriggerSObjectType__c, + UserLicenseDefinitionKey__c = this.logEntry.Log__r.UserLicenseDefinitionKey__c, + UserLicenseName__c = this.logEntry.Log__r.UserLicenseName__c, + UserLoggingLevel__c = this.logEntry.Log__r.UserLoggingLevel__c, + UserLoggingLevelOrdinal__c = this.logEntry.Log__r.UserLoggingLevelOrdinal__c, + UserRoleId__c = this.logEntry.Log__r.UserRoleId__c, + UserRoleName__c = this.logEntry.Log__r.UserRoleName__c, + UserType__c = this.logEntry.Log__r.UserType__c + ); + } + + private void truncateFieldValues() { + for (String fieldName : this.logEntryArchive.getPopulatedFieldsAsMap().keySet()) { + Schema.SObjectField field = Schema.LogEntryArchive__b.SObjectType.getDescribe().fields.getMap().get(fieldName); + if (field.getDescribe().getSoapType() == Schema.SoapType.STRING) { + String fieldValue = (String) this.logEntryArchive.get(field); + this.logEntryArchive.put(field, fieldValue?.left(field.getDescribe().getLength())); + } + } + } +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls new file mode 100644 index 000000000..ef615922d --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls @@ -0,0 +1,863 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.ApexAssertionsShouldIncludeMessage, PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LogEntryArchiveBuilder_Tests { + @IsTest + static void it_should_write_all_fields_over_from_log_entry_event() { + Datetime timestamp = System.now(); + LogEntryEvent__e mockEvent = (LogEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType) + .populateAllFields() + .getRecord(); + mockEvent = (LogEntryEvent__e) LoggerMockDataCreator.setReadOnlyField(mockEvent, Schema.LogEntryEvent__e.EventUuid, 'abc-xyz-some-value'); + mockEvent.LoggingLevel__c = LoggingLevel.INFO.name(); + mockEvent.Timestamp__c = timestamp; + mockEvent.TimestampString__c = String.valueOf(timestamp.getTime()); + + LogEntryArchive__b logEntryArchive = new LogEntryArchiveBuilder(mockEvent).getLogEntryArchive(); + + System.assertNotEquals(null, logEntryArchive); + assertAllFieldsMatch(timestamp, mockEvent, logEntryArchive); + } + + @IsTest + static void it_should_cache_instance_of_log_entry_archive_for_log_entry_event() { + LogEntryEvent__e mockEvent = (LogEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType) + .populateAllFields() + .getRecord(); + mockEvent = (LogEntryEvent__e) LoggerMockDataCreator.setReadOnlyField(mockEvent, Schema.LogEntryEvent__e.EventUuid, 'abc-xyz-some-value'); + LogEntryArchiveBuilder archiveBuilder = new LogEntryArchiveBuilder(mockEvent); + LogEntryArchive__b originalLogEntryArchive = archiveBuilder.getLogEntryArchive(); + originalLogEntryArchive.Message__c = 'Some new value'; + + LogEntryArchive__b secondLogEntryArchive = archiveBuilder.getLogEntryArchive(); + + System.assertEquals(originalLogEntryArchive, secondLogEntryArchive); + } + + @IsTest + static void it_should_truncate_log_entry_archive_field_values_when_too_long() { + LogEntryEvent__e mockEvent = (LogEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType) + .populateAllFields() + .getRecord(); + mockEvent = (LogEntryEvent__e) LoggerMockDataCreator.setReadOnlyField(mockEvent, Schema.LogEntryEvent__e.EventUuid, 'abc-xyz-some-value'); + mockEvent.Message__c = 'Z'.repeat(Schema.LogEntryArchive__b.Message__c.getDescribe().getLength() + 1); + LogEntryArchive__b logEntryArchive = new LogEntryArchiveBuilder(mockEvent).getLogEntryArchive(); + + System.assertNotEquals(mockEvent.Message__c, logEntryArchive.Message__c); + System.assertEquals(mockEvent.Message__c.left(Schema.LogEntryArchive__b.Message__c.getDescribe().getLength()), logEntryArchive.Message__c); + } + + @IsTest + static void it_should_write_all_fields_over_from_log_entry() { + LogEntry__c mockLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType).populateAllFields().getRecord(); + mockLogEntry.Log__r = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateAllFields().getRecord(); + mockLogEntry.LoggingLevel__c = LoggingLevel.INFO.name(); + + LogEntryArchive__b logEntryArchive = new LogEntryArchiveBuilder(mockLogEntry).getLogEntryArchive(); + + System.assertNotEquals(null, logEntryArchive); + assertAllFieldsMatch(mockLogEntry, logEntryArchive); + } + + @IsTest + static void it_should_cache_instance_of_log_entry_archive_for_log_entry() { + LogEntry__c mockLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType).populateRequiredFields().getRecord(); + mockLogEntry.Log__r = (Log__c) LoggerMockDataCreator.createDataBuilder(Schema.Log__c.SObjectType).populateRequiredFields().getRecord(); + LogEntryArchiveBuilder archiveBuilder = new LogEntryArchiveBuilder(mockLogEntry); + LogEntryArchive__b originalLogEntryArchive = archiveBuilder.getLogEntryArchive(); + originalLogEntryArchive.Message__c = 'Some new value'; + + LogEntryArchive__b secondLogEntryArchive = archiveBuilder.getLogEntryArchive(); + + System.assertEquals(originalLogEntryArchive, secondLogEntryArchive); + System.assertEquals(originalLogEntryArchive.Message__c, secondLogEntryArchive.Message__c); + } + + @IsTest + static void it_should_truncate_log_entry_field_values_when_too_long() { + LogEntry__c mockLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType).populateRequiredFields().getRecord(); + mockLogEntry.Message__c = 'Z'.repeat(Schema.LogEntryArchive__b.Message__c.getDescribe().getLength() + 1); + LogEntryArchive__b logEntryArchive = new LogEntryArchiveBuilder(mockLogEntry).getLogEntryArchive(); + + System.assertNotEquals(mockLogEntry.Message__c, logEntryArchive.Message__c); + System.assertEquals(mockLogEntry.Message__c.left(Schema.LogEntryArchive__b.Message__c.getDescribe().getLength()), logEntryArchive.Message__c); + } + + @SuppressWarnings('PMD.NcssMethodCount') + private static void assertAllFieldsMatch(Datetime timestamp, LogEntryEvent__e mockEvent, LogEntryArchive__b logEntryArchive) { + System.assertEquals(mockEvent.ApiVersion__c, logEntryArchive.ApiVersion__c, 'logEntryArchive.ApiVersion__c was not properly set'); + System.assertEquals(UserInfo.getUserId(), logEntryArchive.ArchivedById__c, 'logEntryArchive.ArchivedById__c was not properly set'); + System.assertNotEquals(null, logEntryArchive.ArchivedDate__c, 'logEntryArchive.ArchivedDate__c was not properly set'); + System.assertEquals(System.today(), logEntryArchive.ArchivedDate__c.date(), 'logEntryArchive.ArchivedDate__c was not properly set'); + System.assertEquals(UserInfo.getUsername(), logEntryArchive.ArchivedByUsername__c, 'logEntryArchive.ArchivedByUsername__c was not properly set'); + System.assertEquals(mockEvent.ComponentType__c, logEntryArchive.ComponentType__c, 'logEntryArchive.ComponentType__c was not properly set'); + System.assertEquals( + mockEvent.DatabaseResultCollectionSize__c, + logEntryArchive.DatabaseResultCollectionSize__c, + 'logEntryArchive.DatabaseResultCollectionSize__c was not properly set' + ); + System.assertEquals( + mockEvent.DatabaseResultCollectionType__c, + logEntryArchive.DatabaseResultCollectionType__c, + 'logEntryArchive.DatabaseResultCollectionType__c was not properly set' + ); + System.assertEquals( + mockEvent.DatabaseResultJson__c, + logEntryArchive.DatabaseResultJson__c, + 'logEntryArchive.DatabaseResultJson__c was not properly set' + ); + System.assertEquals( + mockEvent.DatabaseResultType__c, + logEntryArchive.DatabaseResultType__c, + 'logEntryArchive.DatabaseResultType__c was not properly set' + ); + System.assertEquals(mockEvent.EpochTimestamp__c, logEntryArchive.EpochTimestamp__c, 'logEntryArchive.EpochTimestamp__c was not properly set'); + System.assertEquals(mockEvent.EventUuid, logEntryArchive.EventUuid__c, 'logEntryArchive.EventUuid__c was not properly set'); + System.assertEquals(mockEvent.ExceptionMessage__c, logEntryArchive.ExceptionMessage__c, 'logEntryArchive.ExceptionMessage__c was not properly set'); + System.assertEquals( + mockEvent.ExceptionStackTrace__c, + logEntryArchive.ExceptionStackTrace__c, + 'logEntryArchive.ExceptionStackTrace__c was not properly set' + ); + System.assertEquals(mockEvent.ExceptionType__c, logEntryArchive.ExceptionType__c, 'logEntryArchive.ExceptionType__c was not properly set'); + System.assertEquals( + String.valueOf(mockEvent.HttpRequestBody__c), + logEntryArchive.HttpRequestBody__c, + 'logEntryArchive.HttpRequestBody__c was not properly set' + ); + System.assertEquals( + String.valueOf(mockEvent.HttpRequestCompressed__c), + logEntryArchive.HttpRequestCompressed__c, + 'logEntryArchive.HttpRequestCompressed__c was not properly set' + ); + System.assertEquals( + mockEvent.HttpRequestEndpoint__c, + logEntryArchive.HttpRequestEndpoint__c, + 'logEntryArchive.HttpRequestEndpoint__c was not properly set' + ); + System.assertEquals(mockEvent.HttpRequestMethod__c, logEntryArchive.HttpRequestMethod__c, 'logEntryArchive.HttpRequestMethod__c was not properly set'); + System.assertEquals(mockEvent.HttpResponseBody__c, logEntryArchive.HttpResponseBody__c, 'logEntryArchive.HttpResponseBody__c was not properly set'); + System.assertEquals( + mockEvent.HttpResponseHeaderKeys__c, + logEntryArchive.HttpResponseHeaderKeys__c, + 'logEntryArchive.HttpResponseHeaderKeys__c was not properly set' + ); + System.assertEquals( + mockEvent.HttpResponseStatus__c, + logEntryArchive.HttpResponseStatus__c, + 'logEntryArchive.HttpResponseStatus__c was not properly set' + ); + System.assertEquals( + mockEvent.HttpResponseStatusCode__c, + logEntryArchive.HttpResponseStatusCode__c, + 'logEntryArchive.HttpResponseStatusCode__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsAggregateQueriesMax__c, + logEntryArchive.LimitsAggregateQueriesMax__c, + 'logEntryArchive.LimitsAggregateQueriesMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsAggregateQueriesUsed__c, + logEntryArchive.LimitsAggregateQueriesUsed__c, + 'logEntryArchive.LimitsAggregateQueriesUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsAsyncCallsMax__c, + logEntryArchive.LimitsAsyncCallsMax__c, + 'logEntryArchive.LimitsAsyncCallsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsAsyncCallsUsed__c, + logEntryArchive.LimitsAsyncCallsUsed__c, + 'logEntryArchive.LimitsAsyncCallsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsCalloutsUsed__c, + logEntryArchive.LimitsCalloutsUsed__c, + 'logEntryArchive.LimitsCalloutsUsed__c was not properly set' + ); + System.assertEquals(mockEvent.LimitsCpuTimeMax__c, logEntryArchive.LimitsCpuTimeMax__c, 'logEntryArchive.LimitsCpuTimeMax__c was not properly set'); + System.assertEquals(mockEvent.LimitsCpuTimeUsed__c, logEntryArchive.LimitsCpuTimeUsed__c, 'logEntryArchive.LimitsCpuTimeUsed__c was not properly set'); + System.assertEquals(mockEvent.LimitsDmlRowsMax__c, logEntryArchive.LimitsDmlRowsMax__c, 'logEntryArchive.LimitsDmlRowsMax__c was not properly set'); + System.assertEquals(mockEvent.LimitsDmlRowsUsed__c, logEntryArchive.LimitsDmlRowsUsed__c, 'logEntryArchive.LimitsDmlRowsUsed__c was not properly set'); + System.assertEquals( + mockEvent.LimitsDmlStatementsMax__c, + logEntryArchive.LimitsDmlStatementsMax__c, + 'logEntryArchive.LimitsDmlStatementsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsDmlStatementsUsed__c, + logEntryArchive.LimitsDmlStatementsUsed__c, + 'logEntryArchive.LimitsDmlStatementsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsEmailInvocationsMax__c, + logEntryArchive.LimitsEmailInvocationsMax__c, + 'logEntryArchive.LimitsEmailInvocationsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsEmailInvocationsUsed__c, + logEntryArchive.LimitsEmailInvocationsUsed__c, + 'logEntryArchive.LimitsEmailInvocationsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsFutureCallsMax__c, + logEntryArchive.LimitsFutureCallsMax__c, + 'logEntryArchive.LimitsFutureCallsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsFutureCallsUsed__c, + logEntryArchive.LimitsFutureCallsUsed__c, + 'logEntryArchive.LimitsFutureCallsUsed__c was not properly set' + ); + System.assertEquals(mockEvent.LimitsHeapSizeMax__c, logEntryArchive.LimitsHeapSizeMax__c, 'logEntryArchive.LimitsHeapSizeMax__c was not properly set'); + System.assertEquals( + mockEvent.LimitsHeapSizeUsed__c, + logEntryArchive.LimitsHeapSizeUsed__c, + 'logEntryArchive.LimitsHeapSizeUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsPublishImmediateDmlStatementsMax__c, + logEntryArchive.LimitsPublishImmediateDmlStatementsMax__c, + 'logEntryArchive.LimitsPublishImmediateDmlStatementsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsPublishImmediateDmlStatementsUsed__c, + logEntryArchive.LimitsPublishImmediateDmlStatementsUsed__c, + 'logEntryArchive.LimitsPublishImmediateDmlStatementsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsMobilePushApexCallsMax__c, + logEntryArchive.LimitsMobilePushApexCallsMax__c, + 'logEntryArchive.LimitsMobilePushApexCallsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsMobilePushApexCallsUsed__c, + logEntryArchive.LimitsMobilePushApexCallsUsed__c, + 'logEntryArchive.LimitsMobilePushApexCallsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsQueueableJobsMax__c, + logEntryArchive.LimitsQueueableJobsMax__c, + 'logEntryArchive.LimitsQueueableJobsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsQueueableJobsUsed__c, + logEntryArchive.LimitsQueueableJobsUsed__c, + 'logEntryArchive.LimitsQueueableJobsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoqlQueriesMax__c, + logEntryArchive.LimitsSoqlQueriesMax__c, + 'logEntryArchive.LimitsSoqlQueriesMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoqlQueriesUsed__c, + logEntryArchive.LimitsSoqlQueriesUsed__c, + 'logEntryArchive.LimitsSoqlQueriesUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoqlQueryLocatorRowsMax__c, + logEntryArchive.LimitsSoqlQueryLocatorRowsMax__c, + 'logEntryArchive.LimitsSoqlQueryLocatorRowsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoqlQueryLocatorRowsUsed__c, + logEntryArchive.LimitsSoqlQueryLocatorRowsUsed__c, + 'logEntryArchive.LimitsSoqlQueryLocatorRowsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoqlQueryRowsMax__c, + logEntryArchive.LimitsSoqlQueryRowsMax__c, + 'logEntryArchive.LimitsSoqlQueryRowsMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoqlQueryRowsUsed__c, + logEntryArchive.LimitsSoqlQueryRowsUsed__c, + 'logEntryArchive.LimitsSoqlQueryRowsUsed__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoslSearchesMax__c, + logEntryArchive.LimitsSoslSearchesMax__c, + 'logEntryArchive.LimitsSoslSearchesMax__c was not properly set' + ); + System.assertEquals( + mockEvent.LimitsSoslSearchesUsed__c, + logEntryArchive.LimitsSoslSearchesUsed__c, + 'logEntryArchive.LimitsSoslSearchesUsed__c was not properly set' + ); + System.assertEquals(mockEvent.Locale__c, logEntryArchive.Locale__c, 'logEntryArchive.Locale__c was not properly set'); + // System.assertEquals(String.isNotBlank(mockEvent.LoggedById__c) ? mockEvent.LoggedById__c : 'Anonymous', logEntryArchive.LoggedBy__c); + System.assertEquals(mockEvent.LoggedById__c, logEntryArchive.LoggedById__c, 'logEntryArchive.LoggedById__c was not properly set'); + System.assertEquals( + String.isNotBlank(mockEvent.LoggedByUsername__c) ? mockEvent.LoggedByUsername__c : 'Anonymous', + logEntryArchive.LoggedByUsername__c, + 'logEntryArchive.LoggedByUsername__c was not properly set' + ); + System.assertEquals( + mockEvent.LoggerVersionNumber__c, + logEntryArchive.LoggerVersionNumber__c, + 'logEntryArchive.LoggerVersionNumber__c was not properly set' + ); + System.assertEquals(mockEvent.LoggingLevel__c, logEntryArchive.LoggingLevel__c, 'logEntryArchive.LoggingLevel__c was not properly set'); + System.assertEquals( + mockEvent.LoggingLevelOrdinal__c, + logEntryArchive.LoggingLevelOrdinal__c, + 'logEntryArchive.LoggingLevelOrdinal__c was not properly set' + ); + System.assertEquals(mockEvent.LoginApplication__c, logEntryArchive.LoginApplication__c, 'logEntryArchive.LoginApplication__c was not properly set'); + System.assertEquals(mockEvent.LoginBrowser__c, logEntryArchive.LoginBrowser__c, 'logEntryArchive.LoginBrowser__c was not properly set'); + System.assertEquals(mockEvent.LoginHistoryId__c, logEntryArchive.LoginHistoryId__c, 'logEntryArchive.LoginHistoryId__c was not properly set'); + System.assertEquals(mockEvent.LoginPlatform__c, logEntryArchive.LoginPlatform__c, 'logEntryArchive.LoginPlatform__c was not properly set'); + System.assertEquals(mockEvent.LoginType__c, logEntryArchive.LoginType__c, 'logEntryArchive.LoginType__c was not properly set'); + System.assertEquals(mockEvent.LogoutUrl__c, logEntryArchive.LogoutUrl__c, 'logEntryArchive.LogoutUrl__c was not properly set'); + System.assertEquals(mockEvent.Message__c, logEntryArchive.Message__c, 'logEntryArchive.Message__c was not properly set'); + System.assertEquals(mockEvent.NetworkId__c, logEntryArchive.NetworkId__c, 'logEntryArchive.NetworkId__c was not properly set'); + System.assertEquals(mockEvent.NetworkLoginUrl__c, logEntryArchive.NetworkLoginUrl__c, 'logEntryArchive.NetworkLoginUrl__c was not properly set'); + System.assertEquals(mockEvent.NetworkLogoutUrl__c, logEntryArchive.NetworkLogoutUrl__c, 'logEntryArchive.NetworkLogoutUrl__c was not properly set'); + System.assertEquals( + mockEvent.NetworkSelfRegistrationUrl__c, + logEntryArchive.NetworkSelfRegistrationUrl__c, + 'logEntryArchive.NetworkSelfRegistrationUrl__c was not properly set' + ); + System.assertEquals( + mockEvent.NetworkUrlPathPrefix__c, + logEntryArchive.NetworkUrlPathPrefix__c, + 'logEntryArchive.NetworkUrlPathPrefix__c was not properly set' + ); + System.assertEquals( + mockEvent.OrganizationDomainUrl__c, + logEntryArchive.OrganizationDomainUrl__c, + 'logEntryArchive.OrganizationDomainUrl__c was not properly set' + ); + System.assertEquals( + mockEvent.OrganizationEnvironmentType__c, + logEntryArchive.OrganizationEnvironmentType__c, + 'logEntryArchive.OrganizationEnvironmentType__c was not properly set' + ); + System.assertEquals(mockEvent.OrganizationId__c, logEntryArchive.OrganizationId__c, 'logEntryArchive.OrganizationId__c was not properly set'); + System.assertEquals( + mockEvent.OrganizationInstanceName__c, + logEntryArchive.OrganizationInstanceName__c, + 'logEntryArchive.OrganizationInstanceName__c was not properly set' + ); + System.assertEquals(mockEvent.OrganizationName__c, logEntryArchive.OrganizationName__c, 'logEntryArchive.OrganizationName__c was not properly set'); + System.assertEquals( + mockEvent.OrganizationNamespacePrefix__c, + logEntryArchive.OrganizationNamespacePrefix__c, + 'logEntryArchive.OrganizationNamespacePrefix__c was not properly set' + ); + System.assertEquals(mockEvent.OrganizationType__c, logEntryArchive.OrganizationType__c, 'logEntryArchive.OrganizationType__c was not properly set'); + System.assertEquals(mockEvent.OriginLocation__c, logEntryArchive.OriginLocation__c, 'logEntryArchive.OriginLocation__c was not properly set'); + System.assertEquals(mockEvent.OriginType__c, logEntryArchive.OriginType__c, 'logEntryArchive.OriginType__c was not properly set'); + System.assertEquals( + mockEvent.ParentLogTransactionId__c, + logEntryArchive.ParentLogTransactionId__c, + 'logEntryArchive.ParentLogTransactionId__c was not properly set' + ); + System.assertEquals(mockEvent.ProfileId__c, logEntryArchive.ProfileId__c, 'logEntryArchive.ProfileId__c was not properly set'); + System.assertEquals(mockEvent.ProfileName__c, logEntryArchive.ProfileName__c, 'logEntryArchive.ProfileName__c was not properly set'); + System.assertEquals( + mockEvent.RecordCollectionSize__c, + logEntryArchive.RecordCollectionSize__c, + 'logEntryArchive.RecordCollectionSize__c was not properly set' + ); + System.assertEquals( + mockEvent.RecordCollectionType__c, + logEntryArchive.RecordCollectionType__c, + 'logEntryArchive.RecordCollectionType__c was not properly set' + ); + System.assertEquals(mockEvent.RecordId__c, logEntryArchive.RecordId__c, 'logEntryArchive.RecordId__c was not properly set'); + System.assertEquals(mockEvent.RecordJson__c, logEntryArchive.RecordJson__c, 'logEntryArchive.RecordJson__c was not properly set'); + System.assertEquals( + mockEvent.RecordSObjectClassification__c, + logEntryArchive.RecordSObjectClassification__c, + 'logEntryArchive.RecordSObjectClassification__c was not properly set' + ); + System.assertEquals(mockEvent.RecordSObjectType__c, logEntryArchive.RecordSObjectType__c, 'logEntryArchive.RecordSObjectType__c was not properly set'); + System.assertEquals( + mockEvent.RecordSObjectTypeNamespace__c, + logEntryArchive.RecordSObjectTypeNamespace__c, + 'logEntryArchive.RecordSObjectTypeNamespace__c was not properly set' + ); + System.assertEquals(mockEvent.SessionId__c, logEntryArchive.SessionId__c, 'logEntryArchive.SessionId__c was not properly set'); + System.assertEquals( + mockEvent.SessionSecurityLevel__c, + logEntryArchive.SessionSecurityLevel__c, + 'logEntryArchive.SessionSecurityLevel__c was not properly set' + ); + System.assertEquals(mockEvent.SessionType__c, logEntryArchive.SessionType__c, 'logEntryArchive.SessionType__c was not properly set'); + System.assertEquals(mockEvent.SourceIp__c, logEntryArchive.SourceIp__c, 'logEntryArchive.SourceIp__c was not properly set'); + System.assertEquals(mockEvent.StackTrace__c, logEntryArchive.StackTrace__c, 'logEntryArchive.StackTrace__c was not properly set'); + System.assertEquals(mockEvent.SystemMode__c, logEntryArchive.SystemMode__c, 'logEntryArchive.SystemMode__c was not properly set'); + System.assertEquals(mockEvent.Tags__c, logEntryArchive.Tags__c, 'logEntryArchive.Tags__c was not properly set'); + System.assertEquals(mockEvent.ThemeDisplayed__c, logEntryArchive.ThemeDisplayed__c, 'logEntryArchive.ThemeDisplayed__c was not properly set'); + System.assertEquals(timestamp, logEntryArchive.Timestamp__c, 'logEntryArchive.Timestamp__c was not properly set'); + System.assertEquals(String.valueOf(timestamp.getTime()), logEntryArchive.TimestampString__c, 'logEntryArchive.TimestampString__c was not properly set'); + System.assertEquals(mockEvent.TimeZoneId__c, logEntryArchive.TimeZoneId__c, 'logEntryArchive.TimeZoneId__c was not properly set'); + System.assertEquals( + mockEvent.TransactionEntryNumber__c, + logEntryArchive.TransactionEntryNumber__c, + 'logEntryArchive.TransactionEntryNumber__c was not properly set' + ); + System.assertEquals(mockEvent.TransactionId__c, logEntryArchive.TransactionId__c, 'logEntryArchive.TransactionId__c was not properly set'); + System.assertEquals( + mockEvent.TriggerOperationType__c, + logEntryArchive.TriggerOperationType__c, + 'logEntryArchive.TriggerOperationType__c was not properly set' + ); + System.assertEquals( + mockEvent.TriggerSObjectType__c, + logEntryArchive.TriggerSObjectType__c, + 'logEntryArchive.TriggerSObjectType__c was not properly set' + ); + System.assertEquals( + mockEvent.UserLicenseDefinitionKey__c, + logEntryArchive.UserLicenseDefinitionKey__c, + 'logEntryArchive.UserLicenseDefinitionKey__c was not properly set' + ); + System.assertEquals(mockEvent.UserLicenseName__c, logEntryArchive.UserLicenseName__c, 'logEntryArchive.UserLicenseName__c was not properly set'); + System.assertEquals(mockEvent.UserLoggingLevel__c, logEntryArchive.UserLoggingLevel__c, 'logEntryArchive.UserLoggingLevel__c was not properly set'); + System.assertEquals( + mockEvent.UserLoggingLevelOrdinal__c, + logEntryArchive.UserLoggingLevelOrdinal__c, + 'logEntryArchive.UserLoggingLevelOrdinal__c was not properly set' + ); + System.assertEquals(mockEvent.UserRoleId__c, logEntryArchive.UserRoleId__c, 'logEntryArchive.UserRoleId__c was not properly set'); + System.assertEquals(mockEvent.UserRoleName__c, logEntryArchive.UserRoleName__c, 'logEntryArchive.UserRoleName__c was not properly set'); + System.assertEquals(mockEvent.UserType__c, logEntryArchive.UserType__c, 'logEntryArchive.UserType__c was not properly set'); + } + + @SuppressWarnings('PMD.NcssMethodCount') + private static void assertAllFieldsMatch(LogEntry__c logEntry, LogEntryArchive__b logEntryArchive) { + List tagNames = new List(); + for (LogEntryTag__c logEntryTag : logEntry.LogEntryTags__r) { + tagNames.add(logEntryTag.Tag__r.Name); + } + tagNames = new List(new Set(tagNames)); + tagNames.sort(); + String tags = String.join(tagNames, '\n'); + + System.assertEquals( + logEntry.Log__r.ApiReleaseNumber__c, + logEntryArchive.ApiReleaseNumber__c, + 'logEntryArchive.ApiReleaseNumber__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.ApiReleaseVersion__c, + logEntryArchive.ApiReleaseVersion__c, + 'logEntryArchive.ApiReleaseVersion__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.ApiVersion__c, logEntryArchive.ApiVersion__c, 'logEntryArchive.ApiVersion__c was not properly set'); + System.assertEquals(UserInfo.getUserId(), logEntryArchive.ArchivedById__c, 'logEntryArchive.ArchivedById__c was not properly set'); + System.assertNotEquals(null, logEntryArchive.ArchivedDate__c, 'logEntryArchive.ArchivedDate__c was not properly set'); + System.assertEquals(System.today(), logEntryArchive.ArchivedDate__c.date(), 'logEntryArchive.ArchivedDate__c was not properly set'); + System.assertEquals(UserInfo.getUsername(), logEntryArchive.ArchivedByUsername__c, 'logEntryArchive.ArchivedByUsername__c was not properly set'); + System.assertEquals(logEntry.Log__r.ClosedBy__c, logEntryArchive.ClosedById__c, 'logEntryArchive.ClosedById__c was not properly set'); + System.assertEquals( + logEntry.Log__r.ClosedBy__r?.Username, + logEntryArchive.ClosedByUsername__c, + 'logEntryArchive.ClosedByUsername__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.ClosedDate__c, logEntryArchive.ClosedDate__c, 'logEntryArchive.ClosedDate__c was not properly set'); + System.assertEquals(logEntry.Log__r.Comments__c, logEntryArchive.Comments__c, 'logEntryArchive.Comments__c was not properly set'); + System.assertEquals(logEntry.ComponentType__c, logEntryArchive.ComponentType__c, 'logEntryArchive.ComponentType__c was not properly set'); + System.assertEquals( + logEntry.DatabaseResultCollectionSize__c, + logEntryArchive.DatabaseResultCollectionSize__c, + 'logEntryArchive.DatabaseResultCollectionSize__c was not properly set' + ); + System.assertEquals( + logEntry.DatabaseResultCollectionType__c, + logEntryArchive.DatabaseResultCollectionType__c, + 'logEntryArchive.DatabaseResultCollectionType__c was not properly set' + ); + System.assertEquals( + logEntry.DatabaseResultJson__c, + logEntryArchive.DatabaseResultJson__c, + 'logEntryArchive.DatabaseResultJson__c was not properly set' + ); + System.assertEquals( + logEntry.DatabaseResultType__c, + logEntryArchive.DatabaseResultType__c, + 'logEntryArchive.DatabaseResultType__c was not properly set' + ); + System.assertEquals(logEntry.EpochTimestamp__c, logEntryArchive.EpochTimestamp__c, 'logEntryArchive.EpochTimestamp__c was not properly set'); + System.assertEquals(logEntry.EventUuid__c, logEntryArchive.EventUuid__c, 'logEntryArchive.EventUuid__c was not properly set'); + System.assertEquals(logEntry.ExceptionMessage__c, logEntryArchive.ExceptionMessage__c, 'logEntryArchive.ExceptionMessage__c was not properly set'); + System.assertEquals( + logEntry.ExceptionStackTrace__c, + logEntryArchive.ExceptionStackTrace__c, + 'logEntryArchive.ExceptionStackTrace__c was not properly set' + ); + System.assertEquals(logEntry.ExceptionType__c, logEntryArchive.ExceptionType__c, 'logEntryArchive.ExceptionType__c was not properly set'); + System.assertEquals(logEntry.HttpRequestBody__c, logEntryArchive.HttpRequestBody__c, 'logEntryArchive.HttpRequestBody__c was not properly set'); + System.assertEquals( + String.valueOf(logEntry.HttpRequestBodyMasked__c), + logEntryArchive.HttpRequestBodyMasked__c, + 'logEntryArchive.HttpRequestBodyMasked__c was not properly set' + ); + System.assertEquals( + logEntry.HttpRequestEndpoint__c, + logEntryArchive.HttpRequestEndpoint__c, + 'logEntryArchive.HttpRequestEndpoint__c was not properly set' + ); + System.assertEquals(logEntry.HttpRequestMethod__c, logEntryArchive.HttpRequestMethod__c, 'logEntryArchive.HttpRequestMethod__c was not properly set'); + System.assertEquals(logEntry.HttpResponseBody__c, logEntryArchive.HttpResponseBody__c, 'logEntryArchive.HttpResponseBody__c was not properly set'); + System.assertEquals( + String.valueOf(logEntry.HttpResponseBodyMasked__c), + logEntryArchive.HttpResponseBodyMasked__c, + 'logEntryArchive.HttpResponseBodyMasked__c was not properly set' + ); + System.assertEquals( + logEntry.HttpResponseHeaderKeys__c, + logEntryArchive.HttpResponseHeaderKeys__c, + 'logEntryArchive.HttpResponseHeaderKeys__c was not properly set' + ); + System.assertEquals( + logEntry.HttpResponseStatus__c, + logEntryArchive.HttpResponseStatus__c, + 'logEntryArchive.HttpResponseStatus__c was not properly set' + ); + System.assertEquals( + logEntry.HttpResponseStatusCode__c, + logEntryArchive.HttpResponseStatusCode__c, + 'logEntryArchive.HttpResponseStatusCode__c was not properly set' + ); + System.assertEquals(String.valueOf(logEntry.Log__r.IsClosed__c), logEntryArchive.IsClosed__c, 'logEntryArchive.IsClosed__c was not properly set'); + System.assertEquals(String.valueOf(logEntry.Log__r.IsResolved__c), logEntryArchive.IsResolved__c, 'logEntryArchive.IsResolved__c was not properly set'); + System.assertEquals(logEntry.Log__r.Issue__c, logEntryArchive.Issue__c, 'logEntryArchive.Issue__c was not properly set'); + System.assertEquals( + logEntry.LimitsAggregateQueriesMax__c, + logEntryArchive.LimitsAggregateQueriesMax__c, + 'logEntryArchive.LimitsAggregateQueriesMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsAggregateQueriesUsed__c, + logEntryArchive.LimitsAggregateQueriesUsed__c, + 'logEntryArchive.LimitsAggregateQueriesUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsAsyncCallsMax__c, + logEntryArchive.LimitsAsyncCallsMax__c, + 'logEntryArchive.LimitsAsyncCallsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsAsyncCallsUsed__c, + logEntryArchive.LimitsAsyncCallsUsed__c, + 'logEntryArchive.LimitsAsyncCallsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsCalloutsUsed__c, + logEntryArchive.LimitsCalloutsUsed__c, + 'logEntryArchive.LimitsCalloutsUsed__c was not properly set' + ); + System.assertEquals(logEntry.LimitsCpuTimeMax__c, logEntryArchive.LimitsCpuTimeMax__c, 'logEntryArchive.LimitsCpuTimeMax__c was not properly set'); + System.assertEquals(logEntry.LimitsCpuTimeUsed__c, logEntryArchive.LimitsCpuTimeUsed__c, 'logEntryArchive.LimitsCpuTimeUsed__c was not properly set'); + System.assertEquals(logEntry.LimitsDmlRowsMax__c, logEntryArchive.LimitsDmlRowsMax__c, 'logEntryArchive.LimitsDmlRowsMax__c was not properly set'); + System.assertEquals(logEntry.LimitsDmlRowsUsed__c, logEntryArchive.LimitsDmlRowsUsed__c, 'logEntryArchive.LimitsDmlRowsUsed__c was not properly set'); + System.assertEquals( + logEntry.LimitsDmlStatementsMax__c, + logEntryArchive.LimitsDmlStatementsMax__c, + 'logEntryArchive.LimitsDmlStatementsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsDmlStatementsUsed__c, + logEntryArchive.LimitsDmlStatementsUsed__c, + 'logEntryArchive.LimitsDmlStatementsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsEmailInvocationsMax__c, + logEntryArchive.LimitsEmailInvocationsMax__c, + 'logEntryArchive.LimitsEmailInvocationsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsEmailInvocationsUsed__c, + logEntryArchive.LimitsEmailInvocationsUsed__c, + 'logEntryArchive.LimitsEmailInvocationsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsFutureCallsMax__c, + logEntryArchive.LimitsFutureCallsMax__c, + 'logEntryArchive.LimitsFutureCallsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsFutureCallsUsed__c, + logEntryArchive.LimitsFutureCallsUsed__c, + 'logEntryArchive.LimitsFutureCallsUsed__c was not properly set' + ); + System.assertEquals(logEntry.LimitsHeapSizeMax__c, logEntryArchive.LimitsHeapSizeMax__c, 'logEntryArchive.LimitsHeapSizeMax__c was not properly set'); + System.assertEquals( + logEntry.LimitsHeapSizeUsed__c, + logEntryArchive.LimitsHeapSizeUsed__c, + 'logEntryArchive.LimitsHeapSizeUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsMobilePushApexCallsMax__c, + logEntryArchive.LimitsMobilePushApexCallsMax__c, + 'logEntryArchive.LimitsMobilePushApexCallsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsMobilePushApexCallsUsed__c, + logEntryArchive.LimitsMobilePushApexCallsUsed__c, + 'logEntryArchive.LimitsMobilePushApexCallsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsPublishImmediateDmlStatementsMax__c, + logEntryArchive.LimitsPublishImmediateDmlStatementsMax__c, + 'logEntryArchive.LimitsPublishImmediateDmlStatementsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsPublishImmediateDmlStatementsUsed__c, + logEntryArchive.LimitsPublishImmediateDmlStatementsUsed__c, + 'logEntryArchive.LimitsPublishImmediateDmlStatementsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsQueueableJobsMax__c, + logEntryArchive.LimitsQueueableJobsMax__c, + 'logEntryArchive.LimitsQueueableJobsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsQueueableJobsUsed__c, + logEntryArchive.LimitsQueueableJobsUsed__c, + 'logEntryArchive.LimitsQueueableJobsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoqlQueriesMax__c, + logEntryArchive.LimitsSoqlQueriesMax__c, + 'logEntryArchive.LimitsSoqlQueriesMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoqlQueriesUsed__c, + logEntryArchive.LimitsSoqlQueriesUsed__c, + 'logEntryArchive.LimitsSoqlQueriesUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoqlQueryLocatorRowsMax__c, + logEntryArchive.LimitsSoqlQueryLocatorRowsMax__c, + 'logEntryArchive.LimitsSoqlQueryLocatorRowsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoqlQueryLocatorRowsUsed__c, + logEntryArchive.LimitsSoqlQueryLocatorRowsUsed__c, + 'logEntryArchive.LimitsSoqlQueryLocatorRowsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoqlQueryRowsMax__c, + logEntryArchive.LimitsSoqlQueryRowsMax__c, + 'logEntryArchive.LimitsSoqlQueryRowsMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoqlQueryRowsUsed__c, + logEntryArchive.LimitsSoqlQueryRowsUsed__c, + 'logEntryArchive.LimitsSoqlQueryRowsUsed__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoslSearchesMax__c, + logEntryArchive.LimitsSoslSearchesMax__c, + 'logEntryArchive.LimitsSoslSearchesMax__c was not properly set' + ); + System.assertEquals( + logEntry.LimitsSoslSearchesUsed__c, + logEntryArchive.LimitsSoslSearchesUsed__c, + 'logEntryArchive.LimitsSoslSearchesUsed__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.Locale__c, logEntryArchive.Locale__c, 'logEntryArchive.Locale__c was not properly set'); + System.assertEquals(logEntry.Log__r.LoggedBy__c, logEntryArchive.LoggedById__c, 'logEntryArchive.LoggedById__c was not properly set'); + System.assertEquals( + logEntry.Log__r.LoggedByUsername__c, + logEntryArchive.LoggedByUsername__c, + 'logEntryArchive.LoggedByUsername__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.LoggerVersionNumber__c, + logEntryArchive.LoggerVersionNumber__c, + 'logEntryArchive.LoggerVersionNumber__c was not properly set' + ); + System.assertEquals(logEntry.LoggingLevel__c, logEntryArchive.LoggingLevel__c, 'logEntryArchive.LoggingLevel__c was not properly set'); + System.assertEquals( + logEntry.LoggingLevelOrdinal__c, + logEntryArchive.LoggingLevelOrdinal__c, + 'logEntryArchive.LoggingLevelOrdinal__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.LoginApplication__c, + logEntryArchive.LoginApplication__c, + 'logEntryArchive.LoginApplication__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.LoginBrowser__c, logEntryArchive.LoginBrowser__c, 'logEntryArchive.LoginBrowser__c was not properly set'); + System.assertEquals(logEntry.Log__r.LoginHistoryId__c, logEntryArchive.LoginHistoryId__c, 'logEntryArchive.LoginHistoryId__c was not properly set'); + System.assertEquals(logEntry.Log__r.LoginPlatform__c, logEntryArchive.LoginPlatform__c, 'logEntryArchive.LoginPlatform__c was not properly set'); + System.assertEquals(logEntry.Log__r.LoginType__c, logEntryArchive.LoginType__c, 'logEntryArchive.LoginType__c was not properly set'); + System.assertEquals(logEntry.Log__r.LogoutUrl__c, logEntryArchive.LogoutUrl__c, 'logEntryArchive.LogoutUrl__c was not properly set'); + System.assertEquals(logEntry.Log__r.LogPurgeAction__c, logEntryArchive.LogPurgeAction__c, 'logEntryArchive.LogPurgeAction__c was not properly set'); + System.assertEquals( + logEntry.Log__r.LogRetentionDate__c, + logEntryArchive.LogRetentionDate__c, + 'logEntryArchive.LogRetentionDate__c was not properly set' + ); + System.assertEquals(logEntry.Message__c, logEntryArchive.Message__c, 'logEntryArchive.Message__c was not properly set'); + System.assertEquals( + String.valueOf(logEntry.MessageMasked__c), + logEntryArchive.MessageMasked__c, + 'logEntryArchive.MessageMasked__c was not properly set' + ); + System.assertEquals( + String.valueOf(logEntry.MessageTruncated__c), + logEntryArchive.MessageTruncated__c, + 'logEntryArchive.MessageTruncated__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.NetworkId__c, logEntryArchive.NetworkId__c, 'logEntryArchive.NetworkId__c was not properly set'); + System.assertEquals(logEntry.Log__r.NetworkLoginUrl__c, logEntryArchive.NetworkLoginUrl__c, 'logEntryArchive.NetworkLoginUrl__c was not properly set'); + System.assertEquals( + logEntry.Log__r.NetworkLogoutUrl__c, + logEntryArchive.NetworkLogoutUrl__c, + 'logEntryArchive.NetworkLogoutUrl__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.NetworkSelfRegistrationUrl__c, + logEntryArchive.NetworkSelfRegistrationUrl__c, + 'logEntryArchive.NetworkSelfRegistrationUrl__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.NetworkUrlPathPrefix__c, + logEntryArchive.NetworkUrlPathPrefix__c, + 'logEntryArchive.NetworkUrlPathPrefix__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.OrganizationDomainUrl__c, + logEntryArchive.OrganizationDomainUrl__c, + 'logEntryArchive.OrganizationDomainUrl__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.OrganizationEnvironmentType__c, + logEntryArchive.OrganizationEnvironmentType__c, + 'logEntryArchive.OrganizationEnvironmentType__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.OrganizationId__c, logEntryArchive.OrganizationId__c, 'logEntryArchive.OrganizationId__c was not properly set'); + System.assertEquals( + logEntry.Log__r.OrganizationInstanceName__c, + logEntryArchive.OrganizationInstanceName__c, + 'logEntryArchive.OrganizationInstanceName__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.OrganizationName__c, + logEntryArchive.OrganizationName__c, + 'logEntryArchive.OrganizationName__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.OrganizationNamespacePrefix__c, + logEntryArchive.OrganizationNamespacePrefix__c, + 'logEntryArchive.OrganizationNamespacePrefix__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.OrganizationType__c, + logEntryArchive.OrganizationType__c, + 'logEntryArchive.OrganizationType__c was not properly set' + ); + System.assertEquals(logEntry.OriginLocation__c, logEntryArchive.OriginLocation__c, 'logEntryArchive.OriginLocation__c was not properly set'); + System.assertEquals(logEntry.OriginType__c, logEntryArchive.OriginType__c, 'logEntryArchive.OriginType__c was not properly set'); + System.assertEquals( + logEntry.Log__r.ParentLog__r?.TransactionId__c, + logEntryArchive.ParentLogTransactionId__c, + 'logEntryArchive.ParentLogTransactionId__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.Priority__c, logEntryArchive.Priority__c, 'logEntryArchive.Priority__c was not properly set'); + System.assertEquals(logEntry.Log__r.ProfileId__c, logEntryArchive.ProfileId__c, 'logEntryArchive.ProfileId__c was not properly set'); + System.assertEquals(logEntry.Log__r.ProfileName__c, logEntryArchive.ProfileName__c, 'logEntryArchive.ProfileName__c was not properly set'); + System.assertEquals( + logEntry.RecordCollectionSize__c, + logEntryArchive.RecordCollectionSize__c, + 'logEntryArchive.RecordCollectionSize__c was not properly set' + ); + System.assertEquals( + logEntry.RecordCollectionType__c, + logEntryArchive.RecordCollectionType__c, + 'logEntryArchive.RecordCollectionType__c was not properly set' + ); + System.assertEquals(logEntry.RecordId__c, logEntryArchive.RecordId__c, 'logEntryArchive.RecordId__c was not properly set'); + System.assertEquals(logEntry.RecordJson__c, logEntryArchive.RecordJson__c, 'logEntryArchive.RecordJson__c was not properly set'); + System.assertEquals(logEntry.RecordName__c, logEntryArchive.RecordName__c, 'logEntryArchive.RecordName__c was not properly set'); + System.assertEquals( + String.valueOf(logEntry.RecordJsonMasked__c), + logEntryArchive.RecordJsonMasked__c, + 'logEntryArchive.RecordJsonMasked__c was not properly set' + ); + System.assertEquals( + logEntry.RecordSObjectClassification__c, + logEntryArchive.RecordSObjectClassification__c, + 'logEntryArchive.RecordSObjectClassification__c was not properly set' + ); + System.assertEquals(logEntry.RecordSObjectType__c, logEntryArchive.RecordSObjectType__c, 'logEntryArchive.RecordSObjectType__c was not properly set'); + System.assertEquals( + logEntry.RecordSObjectTypeNamespace__c, + logEntryArchive.RecordSObjectTypeNamespace__c, + 'logEntryArchive.RecordSObjectTypeNamespace__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.SessionId__c, logEntryArchive.SessionId__c, 'logEntryArchive.SessionId__c was not properly set'); + System.assertEquals( + logEntry.Log__r.SessionSecurityLevel__c, + logEntryArchive.SessionSecurityLevel__c, + 'logEntryArchive.SessionSecurityLevel__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.SessionType__c, logEntryArchive.SessionType__c, 'logEntryArchive.SessionType__c was not properly set'); + System.assertEquals(logEntry.Log__r.SourceIp__c, logEntryArchive.SourceIp__c, 'logEntryArchive.SourceIp__c was not properly set'); + System.assertEquals(logEntry.StackTrace__c, logEntryArchive.StackTrace__c, 'logEntryArchive.StackTrace__c was not properly set'); + System.assertEquals(logEntry.Log__r.Status__c, logEntryArchive.Status__c, 'logEntryArchive.Status__c was not properly set'); + System.assertEquals(logEntry.Log__r.SystemMode__c, logEntryArchive.SystemMode__c, 'logEntryArchive.SystemMode__c was not properly set'); + System.assertEquals(tags, logEntryArchive.Tags__c, 'logEntryArchive.Tags__c was not properly set'); + System.assertEquals(logEntry.Log__r.ThemeDisplayed__c, logEntryArchive.ThemeDisplayed__c, 'logEntryArchive.ThemeDisplayed__c was not properly set'); + System.assertEquals(logEntry.Timestamp__c, logEntryArchive.Timestamp__c, 'logEntryArchive.Timestamp__c was not properly set'); + System.assertEquals( + String.valueOf(logEntry.Timestamp__c), + logEntryArchive.TimestampString__c, + 'logEntryArchive.TimestampString__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.TimeZoneId__c, logEntryArchive.TimeZoneId__c, 'logEntryArchive.TimeZoneId__c was not properly set'); + System.assertEquals( + logEntry.TransactionEntryNumber__c, + logEntryArchive.TransactionEntryNumber__c, + 'logEntryArchive.TransactionEntryNumber__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.TransactionId__c, logEntryArchive.TransactionId__c, 'logEntryArchive.TransactionId__c was not properly set'); + System.assertEquals( + String.valueOf(logEntry.TriggerIsExecuting__c), + logEntryArchive.TriggerIsExecuting__c, + 'logEntryArchive.TriggerIsExecuting__c was not properly set' + ); + System.assertEquals( + logEntry.TriggerOperationType__c, + logEntryArchive.TriggerOperationType__c, + 'logEntryArchive.TriggerOperationType__c was not properly set' + ); + System.assertEquals( + logEntry.TriggerSObjectType__c, + logEntryArchive.TriggerSObjectType__c, + 'logEntryArchive.TriggerSObjectType__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.UserLicenseDefinitionKey__c, + logEntryArchive.UserLicenseDefinitionKey__c, + 'logEntryArchive.UserLicenseDefinitionKey__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.UserLicenseName__c, logEntryArchive.UserLicenseName__c, 'logEntryArchive.UserLicenseName__c was not properly set'); + System.assertEquals( + logEntry.Log__r.UserLoggingLevel__c, + logEntryArchive.UserLoggingLevel__c, + 'logEntryArchive.UserLoggingLevel__c was not properly set' + ); + System.assertEquals( + logEntry.Log__r.UserLoggingLevelOrdinal__c, + logEntryArchive.UserLoggingLevelOrdinal__c, + 'logEntryArchive.UserLoggingLevelOrdinal__c was not properly set' + ); + System.assertEquals(logEntry.Log__r.UserRoleId__c, logEntryArchive.UserRoleId__c, 'logEntryArchive.UserRoleId__c was not properly set'); + System.assertEquals(logEntry.Log__r.UserRoleName__c, logEntryArchive.UserRoleName__c, 'logEntryArchive.UserRoleName__c was not properly set'); + System.assertEquals(logEntry.Log__r.UserType__c, logEntryArchive.UserType__c, 'logEntryArchive.UserType__c was not properly set'); + } +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveBuilder_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls new file mode 100644 index 000000000..97857448e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls @@ -0,0 +1,102 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Plugins + * @description Controller class used by the LWC `logEntryArchives` to display `LogEntryArchive__b` data + * @see LogEntryArchivePlugin + * @see LogEntryArchiveBuilder + */ +public with sharing class LogEntryArchiveController { + @TestVisible + private static final List MOCK_RECORDS = new List(); + + /** + * @description Returns a list of `LogEntryArchive__b` records, based on the current user's record access + an optional search term for `Message__c` + * @param startDate The initial date to check for matching `LogEntryArchive__b` records, used to filter on `LogEntryArchive__b.Timestamp__c` + * @param endDate The last date to check for matching `LogEntryArchive__b` records, used to filter on `LogEntryArchive__b.Timestamp__c` + * @param rowLimit The max number of rows to return + * @param minimumLoggingLevelOrdinal Optional filter for a minimal logging level ordinal, applied to the field `LoggingLevelOrdinal__c` + * @param messageSearchTerm Optional filter for text contained within the field `Message__c` + * @return The list of matching `LogEntryArchive__b` records + */ + @SuppressWarnings('PMD.CognitiveComplexity, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList') + @AuraEnabled + public static List getLogEntryArchives( + Date startDate, + Date endDate, + Integer rowLimit, + Integer minimumLoggingLevelOrdinal, + String messageSearchTerm + ) { + List matchingLogEntryArchives = new List(); + + // Big Objects have very, very limited support for SOQL filtering, so additional filtering is applied after querying + // Since additional filtering is applied below (outside of the query), so this query gets as many rows + // as possible to maximize the possible matches from post-query filtering + Integer queryLimit = System.Test.isRunningTest() == true ? 1 : Limits.getLimitQueryRows(); + for (LogEntryArchive__b logEntryArchive : [ + SELECT + LoggedByUsername__c, + LoggingLevel__c, + LoggingLevelOrdinal__c, + Message__c, + OriginLocation__c, + OriginType__c, + Tags__c, + Timestamp__c, + TransactionEntryNumber__c, + TransactionId__c + FROM LogEntryArchive__b + WHERE Timestamp__c >= :startDate AND Timestamp__c <= :endDate.addDays(1) + WITH SECURITY_ENFORCED + ORDER BY Timestamp__c DESC, TransactionId__c ASC, TransactionEntryNumber__c DESC + LIMIT :queryLimit + ]) { + addMatches(matchingLogEntryArchives, logEntryArchive, minimumLoggingLevelOrdinal, messageSearchTerm, rowLimit); + } + + // In a test context, creating Big Object data will actually, truly create the data - it's not isolated + // from the org's data. That makes testing trickier, so mock records are injected here. + if (System.Test.isRunningTest() == true) { + matchingLogEntryArchives.clear(); + for (LogEntryArchive__b mockLogEntryArchive : MOCK_RECORDS) { + addMatches(matchingLogEntryArchives, mockLogEntryArchive, minimumLoggingLevelOrdinal, messageSearchTerm, rowLimit); + } + } + + return matchingLogEntryArchives; + } + + @SuppressWarnings('PMD.ExcessiveParameterList') + private static void addMatches( + List matchingLogEntryArchives, + LogEntryArchive__b potentialMatchingArchive, + Integer minimumLoggingLevelOrdinal, + String messageSearchTerm, + Integer rowLimit + ) { + if (matchingLogEntryArchives.size() == rowLimit) { + return; + } + + if (matchesFilterCriteria(potentialMatchingArchive, minimumLoggingLevelOrdinal, messageSearchTerm) == true) { + matchingLogEntryArchives.add(potentialMatchingArchive); + } + } + + private static Boolean matchesFilterCriteria(LogEntryArchive__b logEntryArchive, Integer minimumLoggingLevelOrdinal, String messageSearchTerm) { + Boolean matchesLoggingLevelOrdinal = minimumLoggingLevelOrdinal == null || logEntryArchive.LoggingLevelOrdinal__c >= minimumLoggingLevelOrdinal; + + messageSearchTerm = messageSearchTerm == null ? '' : messageSearchTerm; + String messageValue = logEntryArchive.Message__c == null ? '' : logEntryArchive.Message__c; + Boolean matchesSearchTerm = + String.isBlank(messageSearchTerm) == true || + messageValue.containsIgnoreCase(messageSearchTerm) == true || + Pattern.matches(messageSearchTerm, messageValue) == true; + + return matchesLoggingLevelOrdinal && matchesSearchTerm; + } +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls new file mode 100644 index 000000000..bae74bd6c --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls @@ -0,0 +1,98 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.MethodNamingConventions, PMD.PropertyNamingConventions') +@IsTest(IsParallel=true) +private class LogEntryArchiveController_Tests { + @IsTest + static void it_returns_subset_of_log_entry_archives_when_row_limit_met() { + List expectedLogEntryArchives = createLogEntryArchive(5, LoggingLevel.DEBUG); + LogEntryArchiveController.MOCK_RECORDS.addAll(expectedLogEntryArchives); + Integer rowLimit = expectedLogEntryArchives.size() - 2; + + List returnedLogEntryArchives = LogEntryArchiveController.getLogEntryArchives( + System.today().addDays(-1), + System.today(), + rowLimit, + null, + null + ); + + System.assertNotEquals(0, returnedLogEntryArchives.size(), 'Records should have been returned'); + System.assertEquals(rowLimit, returnedLogEntryArchives.size(), 'Returned records should match the specified row limit'); + } + + @IsTest + static void it_returns_all_log_entry_archives_when_logging_level_and_message_search_term_are_null() { + List expectedLogEntryArchives = createLogEntryArchive(5, LoggingLevel.DEBUG); + LogEntryArchiveController.MOCK_RECORDS.addAll(expectedLogEntryArchives); + + List returnedLogEntryArchives = LogEntryArchiveController.getLogEntryArchives( + System.today().addDays(-1), + System.today(), + 100, + null, + null + ); + + System.assertNotEquals(0, returnedLogEntryArchives.size(), 'Records should have been returned'); + System.assertEquals(expectedLogEntryArchives.size(), returnedLogEntryArchives.size(), 'Returned records should match the expected records'); + } + + @IsTest + static void it_returns_filtered_log_entry_archives_whenlogging_level_filter_is_specified() { + List expectedLogEntryArchives = createLogEntryArchive(5, LoggingLevel.DEBUG); + LoggingLevel minimumLoggingLevel = LoggingLevel.INFO; + expectedLogEntryArchives.get(0).LoggingLevel__c = LoggingLevel.WARN.name(); + expectedLogEntryArchives.get(0).LoggingLevelOrdinal__c = LoggingLevel.WARN.ordinal(); + LogEntryArchiveController.MOCK_RECORDS.addAll(expectedLogEntryArchives); + + List returnedLogEntryArchives = LogEntryArchiveController.getLogEntryArchives( + System.today().addDays(-1), + System.today(), + 100, + minimumLoggingLevel.ordinal(), + null + ); + + System.assertEquals(1, returnedLogEntryArchives.size(), 'Only one matching record should have been returned'); + System.assertEquals(true, returnedLogEntryArchives.get(0).LoggingLevelOrdinal__c > minimumLoggingLevel.ordinal()); + } + + @IsTest + static void it_returns_filtered_log_entry_archives_when_message_search_term_is_specified() { + List expectedLogEntryArchives = createLogEntryArchive(5, LoggingLevel.DEBUG); + String messageSearchTerm = 'some substring'; + expectedLogEntryArchives.get(0).Message__c = 'some extra text and ' + messageSearchTerm + ' and then also some more text, blah blah blah'; + LogEntryArchiveController.MOCK_RECORDS.addAll(expectedLogEntryArchives); + + List returnedLogEntryArchives = LogEntryArchiveController.getLogEntryArchives( + System.today().addDays(-1), + System.today(), + 100, + null, + messageSearchTerm + ); + + System.assertEquals(1, returnedLogEntryArchives.size(), 'Only one matching record should have been returned'); + System.assertEquals(true, returnedLogEntryArchives.get(0).Message__c.contains(messageSearchTerm)); + } + + private static List createLogEntryArchive(Integer numberOfRecordsToCreate, LoggingLevel logLevel) { + List records = new List(); + for (Integer i = 0; i < numberOfRecordsToCreate; i++) { + records.add( + new LogEntryArchive__b( + LoggingLevel__c = logLevel.name(), + LoggingLevelOrdinal__c = logLevel.ordinal(), + Timestamp__c = System.now().addMinutes(-i), + TransactionId__c = null, + TransactionEntryNumber__c = i + 1 + ) + ); + } + return records; + } +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchiveController_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls new file mode 100644 index 000000000..8e70b2f8b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls @@ -0,0 +1,231 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +/** + * @group Plugins + * @description Optional plugin that provides a BigObject, `LogEntryArchive__b`, as an alternative option + * to the platform event `LogEntryEvent__e` + * @see LoggerPlugin + */ +public without sharing class LogEntryArchivePlugin implements LoggerPlugin.Batchable, LoggerPlugin.Triggerable { + @TestVisible + private static final String BIG_OBJECT_IMMEDIATE_LOGGER_SAVE_METHOD = LoggerParameter.getString('CustomSaveMethodBigObjectImmediate', null); + @TestVisible + private static final String BIG_OBJECT_QUEUEABLE_LOGGER_SAVE_METHOD = LoggerParameter.getString('CustomSaveMethodBigObjectQueueable', null); + @TestVisible + private static final String BIG_OBJECT_LOG_PURGE_ACTION = LoggerParameter.getString('CustomLogPurgeActionArchive', null); + @TestVisible + private static final String BIG_OBJECT_STORAGE_LOCATION = LoggerParameter.getString('CustomStorageLocationBigObject', null); + @TestVisible + private static final List LOG_ENTRY_ARCHIVES_TO_SAVE = new List(); + + @TestVisible + private static String lastSaveMethodUsed; + + /** + * @description Default constructor + */ + @SuppressWarnings('PMD.EmptyStatementBlock') + public LogEntryArchivePlugin() { + } + + // Batchable interface methods + /** + * @description Skips directly deleting `LogEntryTag__c` records in `LogBatchPurger` so that the tags + * can be included when `LogEntry__c` records are archived into `LogEntryArchive__b` + * @param configuration The instance of `LoggerPlugin__mdt` configured for this specific plugin + * @param input The instance of `LoggerBatchableContext`, provided by the logging system + */ + public void start(LoggerPlugin__mdt configuration, LoggerBatchableContext input) { + // Skip directly deleting LogEntryTag__c records so that the tags can be included when LogEntry__c records + // are archived into LogEntryArchive__b + if (input.sobjectType == Schema.LogEntryTag__c.SObjectType) { + input.sobjectType = Schema.LogEntry__c.SObjectType; + } + } + + /** + * @description Converts any `LogEntry__c` records into `LogEntryArchive__b` records + * @param configuration The instance of `LoggerPlugin__mdt` configured for this specific plugin + * @param input The instance of `LoggerBatchableContext`, provided by the logging system + * @param loggerRecords The list of `SObject` scope records provider by `LogBatchPurger` + */ + public void execute(LoggerPlugin__mdt configuration, LoggerBatchableContext input, List loggerRecords) { + if (input.sobjectType != Schema.LogEntry__c.SObjectType) { + return; + } + + List logEntries = this.requeryLogEntriesToArchive((List) loggerRecords); + for (LogEntry__c logEntry : logEntries) { + LOG_ENTRY_ARCHIVES_TO_SAVE.add(new LogEntryArchiveBuilder(logEntry).getLogEntryArchive()); + } + this.saveLogEntryArchives(BIG_OBJECT_IMMEDIATE_LOGGER_SAVE_METHOD); + } + + /** + * @description No-op method, required by the interface `LoggerPlugin.Batchable` + * @param configuration The instance of `LoggerPlugin__mdt` configured for this specific plugin + * @param input The instance of `LoggerBatchableContext`, provided by the logging system + */ + @SuppressWarnings('PMD.EmptyStatementBlock') + public void finish(LoggerPlugin__mdt configuration, LoggerBatchableContext input) { + // No-op + } + + // Triggerable interface method + /** + * @description Handles converting Logger's buffer of `LogEntryEvent__e` records into `LogEntryArchive__b` records + * for any user with the included custom save method 'BIG_OBJECT' + * @param configuration The instance of `LoggerPlugin__mdt` configured for this specific plugin + * @param input The instance of `LoggerTriggerableContext`, provided by the logging system + */ + public void execute(LoggerPlugin__mdt configuration, LoggerTriggerableContext input) { + if (input.sobjectType != Schema.LogEntryEvent__e.SObjectType) { + return; + } + Set bigObjectSaveMethods = new Set{ BIG_OBJECT_IMMEDIATE_LOGGER_SAVE_METHOD, BIG_OBJECT_QUEUEABLE_LOGGER_SAVE_METHOD }; + + if ( + input.triggerOperationType == TriggerOperation.BEFORE_INSERT && + bigObjectSaveMethods.contains(Logger.getUserSettings().DefaultSaveMethod__c) == false + ) { + return; + } + + List logEntryEvents = this.filterLogEntryEventsToSave(input); + for (LogEntryEvent__e logEntryEvent : logEntryEvents) { + LOG_ENTRY_ARCHIVES_TO_SAVE.add(new LogEntryArchiveBuilder(logEntryEvent).getLogEntryArchive()); + } + + if (LOG_ENTRY_ARCHIVES_TO_SAVE.isEmpty() == true) { + return; + } + + String saveMethodName; + switch on input.triggerOperationType { + when BEFORE_INSERT { + saveMethodName = Logger.getUserSettings().DefaultSaveMethod__c; + } + when AFTER_INSERT { + saveMethodName = BIG_OBJECT_QUEUEABLE_LOGGER_SAVE_METHOD; + } + } + this.saveLogEntryArchives(saveMethodName); + + Logger.flushBuffer(); + } + + private List requeryLogEntriesToArchive(List logEntries) { + String logLookupRelationshipName = Schema.LogEntry__c.Log__c.getDescribe().getName().removeEnd('__c') + '__r'; + List logFieldsToQuery = new List(); + for (String fieldName : Schema.Log__c.SObjectType.getDescribe().fields.getMap().keySet()) { + logFieldsToQuery.add(logLookupRelationshipName + '.' + fieldName); + } + String closedByUsernameField = getParentFieldPath( + new List{ Schema.LogEntry__c.Log__c, Schema.Log__c.ClosedBy__c, Schema.User.Username } + ); + logFieldsToQuery.add(closedByUsernameField); + String parentLogTransactionIdField = getParentFieldPath( + new List{ Schema.LogEntry__c.Log__c, Schema.Log__c.ParentLog__c, Schema.Log__c.TransactionId__c } + ); + logFieldsToQuery.add(parentLogTransactionIdField); + List logEntryFieldsToQuery = new List(Schema.LogEntry__c.SObjectType.getDescribe().fields.getMap().keySet()); + logEntryFieldsToQuery.addAll(logFieldsToQuery); + + String tagLookupRelationshipName = Schema.LogEntryTag__c.Tag__c.getDescribe().getName().removeEnd('__c') + '__r'; + String tagNameField = tagLookupRelationshipName + '.' + Schema.LoggerTag__c.Name.getDescribe().getName(); + + String logPurgeActionParentFieldName = logLookupRelationshipName + '.' + Schema.Log__c.LogPurgeAction__c.getDescribe().getName(); + String queryTemplate = 'SELECT {0}, (SELECT {1} FROM {2}) FROM {3} WHERE Id IN :logEntries AND {4} = :BIG_OBJECT_LOG_PURGE_ACTION'; + List queryInputs = new List{ + String.join(logEntryFieldsToQuery, ', '), + tagNameField, + 'LogEntryTags__r', // TODO replace with strongly-typed reference or namespace-safe string + Schema.LogEntry__c.SObjectType.getDescribe().getName(), + logPurgeActionParentFieldName + }; + String query = String.escapeSingleQuotes(String.format(queryTemplate, queryInputs)); + return (List) Database.query(query); + } + + private String getParentFieldPath(List parentFieldChainPieces) { + String parentFieldPath = ''; + Integer lastIndex = parentFieldChainPieces.size() - 1; + for (Integer i = 0; i < parentFieldChainPieces.size(); i++) { + Schema.SObjectField field = parentFieldChainPieces.get(i); + if (i != lastIndex) { + parentFieldPath += field.getDescribe().getName().removeEnd('__c') + '__r.'; + } else { + parentFieldPath += field.getDescribe().getName(); + } + } + return parentFieldPath; + } + + private List filterLogEntryEventsToSave(LoggerTriggerableContext input) { + // In a BEFORE_INSERT context, if one of the 2 Big Object save methods is used, then + // all of the LogEntryEvent__e records should be saved in LogEntryArchive__b + if ( + input.triggerOperationType == TriggerOperation.BEFORE_INSERT && + new Set{ BIG_OBJECT_IMMEDIATE_LOGGER_SAVE_METHOD, BIG_OBJECT_QUEUEABLE_LOGGER_SAVE_METHOD } + .contains(Logger.getUserSettings().DefaultSaveMethod__c) + ) { + return (List) input.triggerNew; + } + + if (input.triggerOperationType != TriggerOperation.AFTER_INSERT) { + return new List(); + } + + // In an AFTER_INSERT context, only save LogEntryEvent__e records if the LoggedyId__c user + // as 'BIG_OBJECT' set as their storage location for platform events (DefaultPlatformEventStorageLocation__c) + List logEntryEventsToSave = new List(); + for (LogEntryEvent__e logEntryEvent : (List) input.triggerNew) { + User loggingUser = new User(Id = logEntryEvent.LoggedById__c, ProfileId = logEntryEvent.ProfileId__c); + LoggerSettings__c loggingUserSettings = Logger.getUserSettings(loggingUser); + if (loggingUserSettings.DefaultPlatformEventStorageLocation__c == BIG_OBJECT_STORAGE_LOCATION) { + logEntryEventsToSave.add(logEntryEvent); + } + } + return logEntryEventsToSave; + } + + private void saveLogEntryArchives(String saveMethodName) { + if (System.Test.isRunningTest() == true) { + lastSaveMethodUsed = saveMethodName; + } + + if (saveMethodName == BIG_OBJECT_IMMEDIATE_LOGGER_SAVE_METHOD && System.Test.isRunningTest() == false) { + System.Database.insertImmediate(LOG_ENTRY_ARCHIVES_TO_SAVE); + } else if (saveMethodName == BIG_OBJECT_QUEUEABLE_LOGGER_SAVE_METHOD || saveMethodName == BIG_OBJECT_STORAGE_LOCATION) { + System.enqueueJob(new QueueableSaver(LOG_ENTRY_ARCHIVES_TO_SAVE)); + } + + if (System.Test.isRunningTest() == false) { + LOG_ENTRY_ARCHIVES_TO_SAVE.clear(); + } + } + + /** + * @description Inner class for publishing log entries via the Queueable interface. + */ + private class QueueableSaver implements Queueable { + private List logEntryArchives = new List(); + + private QueueableSaver(List logEntryArchives) { + this.logEntryArchives = logEntryArchives; + } + + /** + * @description Asynchronoulsy inserts the list of `LogEntryArchive_b` records + * @param queueableContext The context of the current queue, provided by the platform + */ + public void execute(System.QueueableContext queueableContext) { + if (System.Test.isRunningTest() == false) { + System.Database.insertImmediate(this.logEntryArchives); + } + } + } +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls new file mode 100644 index 000000000..9f939c478 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls @@ -0,0 +1,239 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.ApexAssertionsShouldIncludeMessage, PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LogEntryArchivePlugin_Tests { + @TestSetup + static void setupData() { + LoggerSObjectHandler.shouldExecute(false); + createParameterConfigurations(); + + Log__c logToArchive = new Log__c( + LogPurgeAction__c = LogEntryArchivePlugin.BIG_OBJECT_LOG_PURGE_ACTION, + LogRetentionDate__c = System.today().addDays(-1), + TransactionId__c = '1234' + ); + Log__c logToDelete = new Log__c(LogPurgeAction__c = 'Delete', LogRetentionDate__c = System.today().addDays(-1), TransactionId__c = '5678'); + insert new List{ logToArchive, logToDelete }; + + List logEntries = new List(); + for (Integer i = 0; i < 10; i++) { + logEntries.add(new LogEntry__c(Log__c = logToArchive.Id, Message__c = 'Some message to archive for ' + i, TransactionEntryNumber__c = i)); + logEntries.add(new LogEntry__c(Log__c = logToDelete.Id, Message__c = 'Some message to skip for ' + i, TransactionEntryNumber__c = i)); + } + insert logEntries; + } + + @IsTest + static void it_archives_log_entry_records_in_log_batch_purger_when_purge_action_is_archive() { + LoggerSObjectHandler.shouldExecute(false); + createParameterConfigurations(); + createPluginConfiguration(); + LogBatchPurger batchPurger = new LogBatchPurger(); + Log__c log = [SELECT Id, (SELECT Id FROM LogEntries__r) FROM Log__c WHERE LogPurgeAction__c = :LogEntryArchivePlugin.BIG_OBJECT_LOG_PURGE_ACTION]; + Test.startTest(); + + Database.executeBatch(batchPurger); + + Test.stopTest(); + System.assertEquals(log.LogEntries__r.size(), LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.size()); + List existingLogs = [SELECT Id FROM Log__c WHERE LogPurgeAction__c = :LogEntryArchivePlugin.BIG_OBJECT_LOG_PURGE_ACTION]; + System.assertEquals(0, existingLogs.size()); + } + + @IsTest + static void it_skips_log_entry_records_in_log_batch_purger_when_purge_action_is_not_archive() { + LoggerSObjectHandler.shouldExecute(false); + createParameterConfigurations(); + createPluginConfiguration(); + LogBatchPurger batchPurger = new LogBatchPurger(); + Log__c logToArchive = [ + SELECT Id, (SELECT Id FROM LogEntries__r) + FROM Log__c + WHERE LogPurgeAction__c = :LogEntryArchivePlugin.BIG_OBJECT_LOG_PURGE_ACTION + ]; + System.assertNotEquals( + true, + logToArchive.LogEntries__r.isEmpty(), + 'Test has started under the wrong condiations - log record should have some related log entries' + ); + Log__c logToSkip = [SELECT Id, (SELECT Id FROM LogEntries__r) FROM Log__c WHERE LogPurgeAction__c = 'Delete']; + System.assertNotEquals( + true, + logToSkip.LogEntries__r.isEmpty(), + 'Test has started under the wrong condiations - log record should have some related log entries' + ); + Test.startTest(); + + Database.executeBatch(batchPurger); + + Test.stopTest(); + System.assertEquals( + logToArchive.LogEntries__r.size(), + LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.size(), + 'Only the log entries for logToArchive should have been archived' + ); + List existingLogs = [SELECT Id FROM Log__c]; + System.assertEquals(0, existingLogs.size(), 'All logs should havve been deleted'); + } + + @IsTest + static void it_includes_tags_for_log_entry_in_log_batch_purger_when_tags_exist() { + LoggerSObjectHandler.shouldExecute(false); + createParameterConfigurations(); + createPluginConfiguration(); + LogBatchPurger batchPurger = new LogBatchPurger(); + Log__c logToArchive = [ + SELECT Id, (SELECT Id FROM LogEntries__r) + FROM Log__c + WHERE LogPurgeAction__c = :LogEntryArchivePlugin.BIG_OBJECT_LOG_PURGE_ACTION + ]; + LoggerTag__c tag = new LoggerTag__c(Name = 'Some tag'); + insert tag; + List logEntryTags = new List(); + for (LogEntry__c logEntry : logToArchive.LogEntries__r) { + logEntryTags.add(new LogEntryTag__c(LogEntry__c = logEntry.Id, Tag__c = tag.Id)); + } + insert logEntryTags; + Test.startTest(); + + Database.executeBatch(batchPurger); + + System.assertEquals(0, Limits.getPublishImmediateDml()); + Test.stopTest(); + System.assertEquals(0, Limits.getPublishImmediateDml()); + System.assertEquals(logToArchive.LogEntries__r.size(), LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.size()); + for (LogEntryArchive__b logEntryArchive : LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE) { + System.assertEquals(tag.Name, logEntryArchive.Tags__c); + } + List existingLogs = [SELECT Id FROM Log__c]; + System.assertEquals(0, existingLogs.size()); + } + + @IsTest + static void it_should_not_create_archive_in_logger_when_save_method_is_not_valid_big_object_save_method() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + createParameterConfigurations(); + createPluginConfiguration(); + Logger.getUserSettings().DefaultSaveMethod__c = Logger.SaveMethod.EVENT_BUS.name(); + LogEntryEvent__e mockEvent = (LogEntryEvent__e) LoggerMockDataCreator.createDataBuilder(Schema.LogEntryEvent__e.SObjectType) + .populateRequiredFields() + .getRecord(); + LoggerPlugin__mdt pluginConfiguration = createPluginConfiguration(); + LoggerTriggerableContext pluginInput = new LoggerTriggerableContext( + Schema.LogEntryEvent__e.SObjectType, + TriggerOperation.BEFORE_INSERT, + new List{ mockEvent }, + null, + null + ); + + new LogEntryArchivePlugin().execute(pluginConfiguration, pluginInput); + + System.assertEquals(0, LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.size(), 'Should not have tried to create big object'); + } + + @IsTest + static void it_should_create_archive_in_logger_when_platform_event_storage_location_is_big_object() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + createParameterConfigurations(); + createPluginConfiguration(); + Logger.getUserSettings().DefaultSaveMethod__c = Logger.SaveMethod.EVENT_BUS.name(); + Logger.getUserSettings().DefaultPlatformEventStorageLocation__c = LogEntryArchivePlugin.BIG_OBJECT_STORAGE_LOCATION; + upsert Logger.getUserSettings(); + System.Test.startTest(); + String transactionId = Logger.getTransactionId(); + Logger.info('Testing big object creation'); + System.assertEquals(1, Logger.getBufferSize(), 'Should have one record in Logger\'s platform event buffer'); + + Logger.saveLog(); + System.Test.getEventBus().deliver(); + + System.assertEquals(0, Logger.getBufferSize(), 'Should have been cleared from Logger\'s platform event buffer'); + System.assertEquals( + 'BIG_OBJECT_QUEUEABLE', + LogEntryArchivePlugin.lastSaveMethodUsed, + 'Last save method used should have been set to BIG_OBJECT_EVENT_BUS' + ); + System.Test.stopTest(); + System.assertEquals(1, LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.size(), 'Should have been put into big object buffer'); + System.assertNotEquals(null, transactionId, 'Should have a value for transaction ID'); + System.assertEquals(transactionId, LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.get(0).TransactionId__c, 'Transaction ID should match'); + System.assertEquals(0, [SELECT COUNT() FROM Log__c WHERE TransactionId__c = :transactionId], 'No custom object data should have been created'); + } + + @IsTest + static void it_should_create_archive_in_logger_when_save_method_is_big_object_immediate() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + createParameterConfigurations(); + createPluginConfiguration(); + Logger.getUserSettings().DefaultSaveMethod__c = LogEntryArchivePlugin.BIG_OBJECT_IMMEDIATE_LOGGER_SAVE_METHOD; + createPluginConfiguration(); + Logger.info('Testing big object creation').getLogEntryEvent(); + + Logger.saveLog(); + + System.assertEquals( + 0, + Limits.getPublishImmediateDml(), + 'Should not have actually used any DML statements in tests because big objects actually get inserted, yikes' + ); + System.assertEquals(0, Logger.getBufferSize(), 'Should not have been put into Logger\'s platform event buffer'); + System.assertEquals(1, LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.size(), 'Should have been put into big object buffer'); + System.assertEquals( + 'BIG_OBJECT_IMMEDIATE', + LogEntryArchivePlugin.lastSaveMethodUsed, + 'Last save method used should have been set to BIG_OBJECT_IMMEDIATE' + ); + } + + @IsTest + static void it_should_create_archive_in_logger_when_save_method_is_big_object_queueable() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + createParameterConfigurations(); + createPluginConfiguration(); + Logger.getUserSettings().DefaultSaveMethod__c = LogEntryArchivePlugin.BIG_OBJECT_QUEUEABLE_LOGGER_SAVE_METHOD; + createPluginConfiguration(); + Logger.info('Testing big object creation').getLogEntryEvent(); + Test.startTest(); + System.assertEquals(0, Limits.getAsyncCalls(), 'Should not have executed any queueable jobs'); + + Logger.saveLog(); + + System.assertEquals(1, Limits.getAsyncCalls(), 'Should have executed queueable saver job'); + System.assertEquals( + 0, + Limits.getPublishImmediateDml(), + 'Should not have actually used any DML statements in tests because big objects actually get inserted, yikes' + ); + System.assertEquals(0, Logger.getBufferSize(), 'Should not have been put into Logger\'s platform event buffer'); + System.assertEquals(1, LogEntryArchivePlugin.LOG_ENTRY_ARCHIVES_TO_SAVE.size(), 'Should have been put into big object buffer'); + System.assertEquals( + 'BIG_OBJECT_QUEUEABLE', + LogEntryArchivePlugin.lastSaveMethodUsed, + 'Last save method used should have been set to BIG_OBJECT_QUEUEABLE' + ); + Test.stopTest(); + } + + private static void createParameterConfigurations() { + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'CustomSaveMethodBigObjectImmediate', Value__c = 'BIG_OBJECT_IMMEDIATE')); + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'CustomSaveMethodBigObjectQueueable', Value__c = 'BIG_OBJECT_QUEUEABLE')); + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'CustomStorageLocationBigObject', Value__c = 'BIG_OBJECT_EVENT_BUS')); + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'CustomLogPurgeActionArchive', Value__c = 'Archive')); + } + + private static LoggerPlugin__mdt createPluginConfiguration() { + LoggerPlugin__mdt configuration = new LoggerPlugin__mdt( + BatchPurgerApexClass__c = LogEntryArchivePlugin.class.getName(), + DeveloperName = 'LogEntryArchivePlugin', + IsEnabled__c = true, + SObjectHandlerApexClass__c = LogEntryArchivePlugin.class.getName() + ); + LoggerTestConfigurator.setMock(configuration); + return configuration; + } +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/classes/LogEntryArchivePlugin_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomLogPurgeActionArchive.md-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomLogPurgeActionArchive.md-meta.xml new file mode 100644 index 000000000..1c3f2a9d0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomLogPurgeActionArchive.md-meta.xml @@ -0,0 +1,17 @@ + + + + true + + Description__c + Adds a new purge action Archive to the list of log purge actions displayed in the LWC loggerSettings + + + Value__c + Archive + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectImmediate.md-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectImmediate.md-meta.xml new file mode 100644 index 000000000..5aa427afa --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectImmediate.md-meta.xml @@ -0,0 +1,17 @@ + + + + true + + Description__c + Adds a new save method BIG_OBJECT_IMMEDIATE to the list of save methods displayed in the LWC loggerSettings + + + Value__c + BIG_OBJECT_IMMEDIATE + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectQueueable.md-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectQueueable.md-meta.xml new file mode 100644 index 000000000..9f6528081 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomSaveMethodBigObjectQueueable.md-meta.xml @@ -0,0 +1,17 @@ + + + + true + + Description__c + Adds a new save method BIG_OBJECT_QUEUEABLE to the list of save methods displayed in the LWC loggerSettings + + + Value__c + BIG_OBJECT_QUEUEABLE + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomStorageLocationBigObject.md-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomStorageLocationBigObject.md-meta.xml new file mode 100644 index 000000000..87ac883c1 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerParameter.CustomStorageLocationBigObject.md-meta.xml @@ -0,0 +1,19 @@ + + + + true + + Description__c + Adds a new storage location BIG_OBJECT_EVENT_BUS to the list of storage locations for platform events (LogEntryEvent__e) + + + Value__c + BIG_OBJECT + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerPlugin.LogEntryArchive.md-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerPlugin.LogEntryArchive.md-meta.xml new file mode 100644 index 000000000..c541e6dc4 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/customMetadata/LoggerPlugin.LogEntryArchive.md-meta.xml @@ -0,0 +1,49 @@ + + + + false + + BatchPurgerApexClass__c + LogEntryArchivePlugin + + + BatchPurgerExecutionOrder__c + + + + BatchPurgerFlowName__c + + + + Description__c + Adds the ability to log directly to the BigObject LogEntryArchive__b + + + IsEnabled__c + true + + + Link__c + + + + SObjectHandlerApexClass__c + LogEntryArchivePlugin + + + SObjectHandlerExecutionOrder__c + + + + SObjectHandlerFlowName__c + + + + VersionNumber__c + + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/data/LoggerSObjectMetadata.getSchemaForName.json b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/data/LoggerSObjectMetadata.getSchemaForName.json new file mode 100644 index 000000000..2b7463634 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/data/LoggerSObjectMetadata.getSchemaForName.json @@ -0,0 +1,799 @@ +{ + "apiName": "LogEntryArchive__b", + "fields": { + "Id": { + "apiName": "Id", + "label": "ID", + "localApiName": "Id", + "type": "id" + }, + "CreatedById": { + "apiName": "CreatedById", + "label": "Created By ID", + "localApiName": "CreatedById", + "type": "reference" + }, + "CreatedDate": { + "apiName": "CreatedDate", + "label": "Created Date", + "localApiName": "CreatedDate", + "type": "datetime" + }, + "SystemModstamp": { + "apiName": "SystemModstamp", + "label": "System Modstamp", + "localApiName": "SystemModstamp", + "type": "datetime" + }, + "ApiReleaseNumber__c": { + "apiName": "ApiReleaseNumber__c", + "inlineHelpText": "The release number for the org's instance - determined by making a callout to status.salesforce.com", + "label": "API Release Number", + "localApiName": "ApiReleaseNumber__c", + "type": "string" + }, + "ApiReleaseVersion__c": { + "apiName": "ApiReleaseVersion__c", + "inlineHelpText": "The release version for the org's instance - determined by making a callout to status.salesforce.com", + "label": "API Release Version", + "localApiName": "ApiReleaseVersion__c", + "type": "string" + }, + "ApiVersion__c": { + "apiName": "ApiVersion__c", + "label": "API Version", + "localApiName": "ApiVersion__c", + "type": "string" + }, + "ClosedById__c": { + "apiName": "ClosedById__c", + "label": "Closed By ID", + "localApiName": "ClosedById__c", + "type": "string" + }, + "ClosedByUsername__c": { + "apiName": "ClosedByUsername__c", + "label": "Closed By Username", + "localApiName": "ClosedByUsername__c", + "type": "string" + }, + "ClosedDate__c": { + "apiName": "ClosedDate__c", + "label": "Closed Date", + "localApiName": "ClosedDate__c", + "type": "datetime" + }, + "Comments__c": { + "apiName": "Comments__c", + "label": "Comments", + "localApiName": "Comments__c", + "type": "textarea" + }, + "ComponentType__c": { + "apiName": "ComponentType__c", + "label": "Component Type", + "localApiName": "ComponentType__c", + "type": "string" + }, + "DatabaseResultCollectionSize__c": { + "apiName": "DatabaseResultCollectionSize__c", + "inlineHelpText": "The number of items contained in the collection of database results", + "label": "Database Result Collection Size", + "localApiName": "DatabaseResultCollectionSize__c", + "type": "double" + }, + "DatabaseResultCollectionType__c": { + "apiName": "DatabaseResultCollectionType__c", + "label": "Database Result Collection Type", + "localApiName": "DatabaseResultCollectionType__c", + "type": "string" + }, + "DatabaseResultJson__c": { + "apiName": "DatabaseResultJson__c", + "label": "Database Result JSON", + "localApiName": "DatabaseResultJson__c", + "type": "textarea" + }, + "DatabaseResultType__c": { + "apiName": "DatabaseResultType__c", + "label": "Database Result Type", + "localApiName": "DatabaseResultType__c", + "type": "string" + }, + "EpochTimestamp__c": { + "apiName": "EpochTimestamp__c", + "label": "Epoch Timestamp", + "localApiName": "EpochTimestamp__c", + "type": "double" + }, + "ExceptionMessage__c": { + "apiName": "ExceptionMessage__c", + "label": "Exception Message", + "localApiName": "ExceptionMessage__c", + "type": "textarea" + }, + "ExceptionStackTrace__c": { + "apiName": "ExceptionStackTrace__c", + "label": "Exception Stack Trace", + "localApiName": "ExceptionStackTrace__c", + "type": "textarea" + }, + "ExceptionType__c": { + "apiName": "ExceptionType__c", + "label": "Exception Type", + "localApiName": "ExceptionType__c", + "type": "string" + }, + "IsClosed__c": { + "apiName": "IsClosed__c", + "label": "Is Closed", + "localApiName": "IsClosed__c", + "type": "string" + }, + "IsResolved__c": { + "apiName": "IsResolved__c", + "label": "Is Resolved", + "localApiName": "IsResolved__c", + "type": "string" + }, + "Issue__c": { + "apiName": "Issue__c", + "label": "Issue", + "localApiName": "Issue__c", + "type": "string" + }, + "LimitsAggregateQueriesMax__c": { + "apiName": "LimitsAggregateQueriesMax__c", + "label": "Aggregate Queries Max", + "localApiName": "LimitsAggregateQueriesMax__c", + "type": "double" + }, + "LimitsAggregateQueriesUsed__c": { + "apiName": "LimitsAggregateQueriesUsed__c", + "label": "Aggregate Queries Used", + "localApiName": "LimitsAggregateQueriesUsed__c", + "type": "double" + }, + "LimitsAggregateQueryMax__c": { + "apiName": "LimitsAggregateQueryMax__c", + "label": "Limits AggregateQueryMax_", + "localApiName": "LimitsAggregateQueryMax__c", + "type": "double" + }, + "LimitsAsyncCallsMax__c": { + "apiName": "LimitsAsyncCallsMax__c", + "label": "Async Calls Max", + "localApiName": "LimitsAsyncCallsMax__c", + "type": "double" + }, + "LimitsAsyncCallsUsed__c": { + "apiName": "LimitsAsyncCallsUsed__c", + "label": "Async Calls Used", + "localApiName": "LimitsAsyncCallsUsed__c", + "type": "double" + }, + "LimitsCalloutsMax__c": { + "apiName": "LimitsCalloutsMax__c", + "label": "Callouts Max", + "localApiName": "LimitsCalloutsMax__c", + "type": "double" + }, + "LimitsCalloutsUsed__c": { + "apiName": "LimitsCalloutsUsed__c", + "label": "Callouts Used", + "localApiName": "LimitsCalloutsUsed__c", + "type": "double" + }, + "LimitsCpuTimeMax__c": { + "apiName": "LimitsCpuTimeMax__c", + "label": "CPU Time Max", + "localApiName": "LimitsCpuTimeMax__c", + "type": "double" + }, + "LimitsCpuTimeUsed__c": { + "apiName": "LimitsCpuTimeUsed__c", + "label": "CPU Time Used", + "localApiName": "LimitsCpuTimeUsed__c", + "type": "double" + }, + "LimitsDmlRowsMax__c": { + "apiName": "LimitsDmlRowsMax__c", + "label": "DML Rows Max", + "localApiName": "LimitsDmlRowsMax__c", + "type": "double" + }, + "LimitsDmlRowsUsed__c": { + "apiName": "LimitsDmlRowsUsed__c", + "label": "DML Rows Used", + "localApiName": "LimitsDmlRowsUsed__c", + "type": "double" + }, + "LimitsDmlStatementsMax__c": { + "apiName": "LimitsDmlStatementsMax__c", + "label": "DML Statements Max", + "localApiName": "LimitsDmlStatementsMax__c", + "type": "double" + }, + "LimitsDmlStatementsUsed__c": { + "apiName": "LimitsDmlStatementsUsed__c", + "label": "DML Statements Used", + "localApiName": "LimitsDmlStatementsUsed__c", + "type": "double" + }, + "LimitsEmailInvocationsMax__c": { + "apiName": "LimitsEmailInvocationsMax__c", + "label": "Email Invocations Max", + "localApiName": "LimitsEmailInvocationsMax__c", + "type": "double" + }, + "LimitsEmailInvocationsUsed__c": { + "apiName": "LimitsEmailInvocationsUsed__c", + "label": "Email Invocations Used", + "localApiName": "LimitsEmailInvocationsUsed__c", + "type": "double" + }, + "LimitsFutureCallsMax__c": { + "apiName": "LimitsFutureCallsMax__c", + "label": "Future Calls Max", + "localApiName": "LimitsFutureCallsMax__c", + "type": "double" + }, + "LimitsFutureCallsUsed__c": { + "apiName": "LimitsFutureCallsUsed__c", + "label": "Future Calls Used", + "localApiName": "LimitsFutureCallsUsed__c", + "type": "double" + }, + "LimitsHeapSizeMax__c": { + "apiName": "LimitsHeapSizeMax__c", + "label": "Heap Size Max", + "localApiName": "LimitsHeapSizeMax__c", + "type": "double" + }, + "LimitsHeapSizeUsed__c": { + "apiName": "LimitsHeapSizeUsed__c", + "label": "Heap Size Used", + "localApiName": "LimitsHeapSizeUsed__c", + "type": "double" + }, + "LimitsMobilePushApexCallsMax__c": { + "apiName": "LimitsMobilePushApexCallsMax__c", + "label": "Mobile Push Apex Calls Max", + "localApiName": "LimitsMobilePushApexCallsMax__c", + "type": "double" + }, + "LimitsMobilePushApexCallsUsed__c": { + "apiName": "LimitsMobilePushApexCallsUsed__c", + "label": "Mobile Push Apex Calls Used", + "localApiName": "LimitsMobilePushApexCallsUsed__c", + "type": "double" + }, + "LimitsPublishImmediateDmlStatementsMax__c": { + "apiName": "LimitsPublishImmediateDmlStatementsMax__c", + "label": "Publish Immediate Statements DML Max", + "localApiName": "LimitsPublishImmediateDmlStatementsMax__c", + "type": "double" + }, + "LimitsPublishImmediateDmlStatementsUsed__c": { + "apiName": "LimitsPublishImmediateDmlStatementsUsed__c", + "label": "Publish Immediate Statements DML Used", + "localApiName": "LimitsPublishImmediateDmlStatementsUsed__c", + "type": "double" + }, + "LimitsQueueableJobsMax__c": { + "apiName": "LimitsQueueableJobsMax__c", + "label": "Queueable Jobs Max", + "localApiName": "LimitsQueueableJobsMax__c", + "type": "double" + }, + "LimitsQueueableJobsUsed__c": { + "apiName": "LimitsQueueableJobsUsed__c", + "label": "Queueable Jobs Used", + "localApiName": "LimitsQueueableJobsUsed__c", + "type": "double" + }, + "LimitsSoqlQueriesMax__c": { + "apiName": "LimitsSoqlQueriesMax__c", + "label": "SOQL Queries Max", + "localApiName": "LimitsSoqlQueriesMax__c", + "type": "double" + }, + "LimitsSoqlQueriesUsed__c": { + "apiName": "LimitsSoqlQueriesUsed__c", + "label": "SOQL Queries Used", + "localApiName": "LimitsSoqlQueriesUsed__c", + "type": "double" + }, + "LimitsSoqlQueryLocatorRowsMax__c": { + "apiName": "LimitsSoqlQueryLocatorRowsMax__c", + "label": "SOQL Query Locator Rows Max", + "localApiName": "LimitsSoqlQueryLocatorRowsMax__c", + "type": "double" + }, + "LimitsSoqlQueryLocatorRowsUsed__c": { + "apiName": "LimitsSoqlQueryLocatorRowsUsed__c", + "label": "SOQL Query Locator Rows Used", + "localApiName": "LimitsSoqlQueryLocatorRowsUsed__c", + "type": "double" + }, + "LimitsSoqlQueryRowsMax__c": { + "apiName": "LimitsSoqlQueryRowsMax__c", + "label": "SOQL Query Rows Max", + "localApiName": "LimitsSoqlQueryRowsMax__c", + "type": "double" + }, + "LimitsSoqlQueryRowsUsed__c": { + "apiName": "LimitsSoqlQueryRowsUsed__c", + "label": "SOQL Query Rows Used", + "localApiName": "LimitsSoqlQueryRowsUsed__c", + "type": "double" + }, + "LimitsSoslSearchesMax__c": { + "apiName": "LimitsSoslSearchesMax__c", + "label": "SOSL Searches Max", + "localApiName": "LimitsSoslSearchesMax__c", + "type": "double" + }, + "LimitsSoslSearchesUsed__c": { + "apiName": "LimitsSoslSearchesUsed__c", + "label": "SOSL Searches Used", + "localApiName": "LimitsSoslSearchesUsed__c", + "type": "double" + }, + "Locale__c": { + "apiName": "Locale__c", + "label": "Locale", + "localApiName": "Locale__c", + "type": "string" + }, + "LogEntryName__c": { + "apiName": "LogEntryName__c", + "label": "Log Name", + "localApiName": "LogEntryName__c", + "type": "string" + }, + "LogName__c": { + "apiName": "LogName__c", + "label": "Log Name", + "localApiName": "LogName__c", + "type": "string" + }, + "LogPurgeAction__c": { + "apiName": "LogPurgeAction__c", + "label": "Log Purge Action", + "localApiName": "LogPurgeAction__c", + "type": "string" + }, + "LogRetentionDate__c": { + "apiName": "LogRetentionDate__c", + "label": "Log Retention Date", + "localApiName": "LogRetentionDate__c", + "type": "datetime" + }, + "LoggedById__c": { + "apiName": "LoggedById__c", + "label": "Logged By ID", + "localApiName": "LoggedById__c", + "type": "string" + }, + "LoggedByUsername__c": { + "apiName": "LoggedByUsername__c", + "label": "Username", + "localApiName": "LoggedByUsername__c", + "type": "string" + }, + "LoggerVersionNumber__c": { + "apiName": "LoggerVersionNumber__c", + "label": "Logger Version Number", + "localApiName": "LoggerVersionNumber__c", + "type": "string" + }, + "LoggingLevelOrdinal__c": { + "apiName": "LoggingLevelOrdinal__c", + "label": "Logging Level Ordinal", + "localApiName": "LoggingLevelOrdinal__c", + "type": "double" + }, + "LoggingLevel__c": { + "apiName": "LoggingLevel__c", + "label": "Logging Level", + "localApiName": "LoggingLevel__c", + "type": "string" + }, + "LoginApplication__c": { + "apiName": "LoginApplication__c", + "label": "Login Application", + "localApiName": "LoginApplication__c", + "type": "string" + }, + "LoginBrowser__c": { + "apiName": "LoginBrowser__c", + "label": "Login Browser", + "localApiName": "LoginBrowser__c", + "type": "string" + }, + "LoginDomain__c": { + "apiName": "LoginDomain__c", + "label": "DEPRECATED: Login Domain", + "localApiName": "LoginDomain__c", + "type": "string" + }, + "LoginHistoryId__c": { + "apiName": "LoginHistoryId__c", + "label": "Login History ID", + "localApiName": "LoginHistoryId__c", + "type": "string" + }, + "LoginPlatform__c": { + "apiName": "LoginPlatform__c", + "label": "Login Platform", + "localApiName": "LoginPlatform__c", + "type": "string" + }, + "LoginType__c": { + "apiName": "LoginType__c", + "label": "Login Type", + "localApiName": "LoginType__c", + "type": "string" + }, + "LogoutUrl__c": { + "apiName": "LogoutUrl__c", + "label": "Logout URL", + "localApiName": "LogoutUrl__c", + "type": "string" + }, + "MessageMasked__c": { + "apiName": "MessageMasked__c", + "label": "Message Masked", + "localApiName": "MessageMasked__c", + "type": "string" + }, + "MessageTruncated__c": { + "apiName": "MessageTruncated__c", + "label": "Message Truncated", + "localApiName": "MessageTruncated__c", + "type": "string" + }, + "Message__c": { + "apiName": "Message__c", + "label": "Message", + "localApiName": "Message__c", + "type": "textarea" + }, + "NetworkId__c": { + "apiName": "NetworkId__c", + "label": "Network ID", + "localApiName": "NetworkId__c", + "type": "string" + }, + "NetworkLoginUrl__c": { + "apiName": "NetworkLoginUrl__c", + "label": "Site Login URL", + "localApiName": "NetworkLoginUrl__c", + "type": "string" + }, + "NetworkLogoutUrl__c": { + "apiName": "NetworkLogoutUrl__c", + "label": "Site Logout URL", + "localApiName": "NetworkLogoutUrl__c", + "type": "string" + }, + "NetworkName__c": { + "apiName": "NetworkName__c", + "label": "Site Name", + "localApiName": "NetworkName__c", + "type": "string" + }, + "NetworkSelfRegistrationUrl__c": { + "apiName": "NetworkSelfRegistrationUrl__c", + "label": "Site Self Registration URL", + "localApiName": "NetworkSelfRegistrationUrl__c", + "type": "string" + }, + "NetworkUrlPathPrefix__c": { + "apiName": "NetworkUrlPathPrefix__c", + "label": "Site URL Path Prefix", + "localApiName": "NetworkUrlPathPrefix__c", + "type": "string" + }, + "OrganizationDomainUrl__c": { + "apiName": "OrganizationDomainUrl__c", + "label": "Organization Domain URL", + "localApiName": "OrganizationDomainUrl__c", + "type": "string" + }, + "OrganizationEnvironmentType__c": { + "apiName": "OrganizationEnvironmentType__c", + "label": "Organization Environment Type", + "localApiName": "OrganizationEnvironmentType__c", + "type": "string" + }, + "OrganizationId__c": { + "apiName": "OrganizationId__c", + "label": "Organization ID", + "localApiName": "OrganizationId__c", + "type": "string" + }, + "OrganizationInstanceName__c": { + "apiName": "OrganizationInstanceName__c", + "label": "Organization Instance Name", + "localApiName": "OrganizationInstanceName__c", + "type": "string" + }, + "OrganizationInstanceReleaseCycle__c": { + "apiName": "OrganizationInstanceReleaseCycle__c", + "label": "DEPRECATED: Instance Release Cycle", + "localApiName": "OrganizationInstanceReleaseCycle__c", + "type": "string" + }, + "OrganizationName__c": { + "apiName": "OrganizationName__c", + "label": "Organization Name", + "localApiName": "OrganizationName__c", + "type": "string" + }, + "OrganizationNamespacePrefix__c": { + "apiName": "OrganizationNamespacePrefix__c", + "label": "Organization Namespace Prefix", + "localApiName": "OrganizationNamespacePrefix__c", + "type": "string" + }, + "OrganizationType__c": { + "apiName": "OrganizationType__c", + "label": "Organization Type", + "localApiName": "OrganizationType__c", + "type": "string" + }, + "OriginLocation__c": { + "apiName": "OriginLocation__c", + "label": "Origin Location", + "localApiName": "OriginLocation__c", + "type": "string" + }, + "OriginType__c": { + "apiName": "OriginType__c", + "label": "Origin Type", + "localApiName": "OriginType__c", + "type": "string" + }, + "ParentLogTransactionId__c": { + "apiName": "ParentLogTransactionId__c", + "label": "Parent Log Transaction ID", + "localApiName": "ParentLogTransactionId__c", + "type": "string" + }, + "Priority__c": { + "apiName": "Priority__c", + "label": "Priority", + "localApiName": "Priority__c", + "type": "string" + }, + "ProfileId__c": { + "apiName": "ProfileId__c", + "label": "Profile ID", + "localApiName": "ProfileId__c", + "type": "string" + }, + "ProfileName__c": { + "apiName": "ProfileName__c", + "label": "Profile Name", + "localApiName": "ProfileName__c", + "type": "string" + }, + "RecordCollectionSize__c": { + "apiName": "RecordCollectionSize__c", + "inlineHelpText": "The number of items contained in the collection of database results", + "label": "Related Record Collection Size", + "localApiName": "RecordCollectionSize__c", + "type": "double" + }, + "RecordCollectionType__c": { + "apiName": "RecordCollectionType__c", + "label": "Related Record Collection Type", + "localApiName": "RecordCollectionType__c", + "type": "string" + }, + "RecordId__c": { + "apiName": "RecordId__c", + "label": "Related Record ID", + "localApiName": "RecordId__c", + "type": "string" + }, + "RecordJsonMasked__c": { + "apiName": "RecordJsonMasked__c", + "label": "Record JSON Masked", + "localApiName": "RecordJsonMasked__c", + "type": "string" + }, + "RecordJson__c": { + "apiName": "RecordJson__c", + "label": "Related Record JSON", + "localApiName": "RecordJson__c", + "type": "textarea" + }, + "RecordSObjectClassification__c": { + "apiName": "RecordSObjectClassification__c", + "label": "Related Record SObject Classification", + "localApiName": "RecordSObjectClassification__c", + "type": "string" + }, + "RecordSObjectTypeNamespace__c": { + "apiName": "RecordSObjectTypeNamespace__c", + "label": "Related Record Object Namespace", + "localApiName": "RecordSObjectTypeNamespace__c", + "type": "string" + }, + "RecordSObjectType__c": { + "apiName": "RecordSObjectType__c", + "label": "Related Record SObject Type", + "localApiName": "RecordSObjectType__c", + "type": "string" + }, + "Scenario__c": { + "apiName": "Scenario__c", + "label": "Scenario", + "localApiName": "Scenario__c", + "type": "string" + }, + "SessionId__c": { + "apiName": "SessionId__c", + "label": "SessionId", + "localApiName": "SessionId__c", + "type": "string" + }, + "SessionSecurityLevel__c": { + "apiName": "SessionSecurityLevel__c", + "label": "Session Security Level", + "localApiName": "SessionSecurityLevel__c", + "type": "string" + }, + "SessionType__c": { + "apiName": "SessionType__c", + "label": "Session Type", + "localApiName": "SessionType__c", + "type": "string" + }, + "SourceIp__c": { + "apiName": "SourceIp__c", + "label": "Source IP", + "localApiName": "SourceIp__c", + "type": "string" + }, + "StackTrace__c": { + "apiName": "StackTrace__c", + "label": "Stack Trace", + "localApiName": "StackTrace__c", + "type": "textarea" + }, + "Status__c": { + "apiName": "Status__c", + "label": "Status", + "localApiName": "Status__c", + "type": "string" + }, + "SystemMode__c": { + "apiName": "SystemMode__c", + "label": "System Mode", + "localApiName": "SystemMode__c", + "type": "string" + }, + "Tags__c": { + "apiName": "Tags__c", + "label": "Tags", + "localApiName": "Tags__c", + "type": "textarea" + }, + "ThemeDisplayed__c": { + "apiName": "ThemeDisplayed__c", + "label": "Theme Displayed", + "localApiName": "ThemeDisplayed__c", + "type": "string" + }, + "TimeZoneId__c": { + "apiName": "TimeZoneId__c", + "label": "Time Zone ID", + "localApiName": "TimeZoneId__c", + "type": "string" + }, + "TimeZoneName__c": { + "apiName": "TimeZoneName__c", + "label": "Time Zone Name", + "localApiName": "TimeZoneName__c", + "type": "string" + }, + "TimestampString__c": { + "apiName": "TimestampString__c", + "label": "Timestamp String", + "localApiName": "TimestampString__c", + "type": "string" + }, + "Timestamp__c": { + "apiName": "Timestamp__c", + "label": "Timestamp", + "localApiName": "Timestamp__c", + "type": "datetime" + }, + "TransactionEntryNumber__c": { + "apiName": "TransactionEntryNumber__c", + "label": "Entry #", + "localApiName": "TransactionEntryNumber__c", + "type": "double" + }, + "TransactionId__c": { + "apiName": "TransactionId__c", + "label": "Transaction ID", + "localApiName": "TransactionId__c", + "type": "string" + }, + "TriggerIsExecuting__c": { + "apiName": "TriggerIsExecuting__c", + "label": "Trigger Is Executing", + "localApiName": "TriggerIsExecuting__c", + "type": "string" + }, + "TriggerOperationType__c": { + "apiName": "TriggerOperationType__c", + "label": "Trigger Operation Type", + "localApiName": "TriggerOperationType__c", + "type": "string" + }, + "TriggerSObjectType__c": { + "apiName": "TriggerSObjectType__c", + "label": "Trigger SObject Type", + "localApiName": "TriggerSObjectType__c", + "type": "string" + }, + "UserLicenseDefinitionKey__c": { + "apiName": "UserLicenseDefinitionKey__c", + "label": "User License Definition Key", + "localApiName": "UserLicenseDefinitionKey__c", + "type": "string" + }, + "UserLicenseId__c": { + "apiName": "UserLicenseId__c", + "label": "User License ID", + "localApiName": "UserLicenseId__c", + "type": "string" + }, + "UserLicenseName__c": { + "apiName": "UserLicenseName__c", + "label": "User License Name", + "localApiName": "UserLicenseName__c", + "type": "string" + }, + "UserLoggingLevelOrdinal__c": { + "apiName": "UserLoggingLevelOrdinal__c", + "label": "User Logging Level Ordinal", + "localApiName": "UserLoggingLevelOrdinal__c", + "type": "double" + }, + "UserLoggingLevel__c": { + "apiName": "UserLoggingLevel__c", + "label": "User Logging Level", + "localApiName": "UserLoggingLevel__c", + "type": "string" + }, + "UserRoleId__c": { + "apiName": "UserRoleId__c", + "label": "User Role ID", + "localApiName": "UserRoleId__c", + "type": "string" + }, + "UserRoleName__c": { + "apiName": "UserRoleName__c", + "label": "User Role Name", + "localApiName": "UserRoleName__c", + "type": "string" + }, + "UserType__c": { + "apiName": "UserType__c", + "label": "User Type", + "localApiName": "UserType__c", + "type": "string" + } + }, + "label": "Log Entry Archive", + "labelPlural": "Log Entry Archives", + "localApiName": "LogEntryArchive__b", + "namespacePrefix": "" +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/logEntryArchives.test.js b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/logEntryArchives.test.js new file mode 100644 index 000000000..02ef174a2 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/__tests__/logEntryArchives.test.js @@ -0,0 +1,261 @@ +import { createElement } from 'lwc'; +import LogEntryArchives from 'c/logEntryArchives'; +import getLogEntryArchives from '@salesforce/apex/LogEntryArchiveController.getLogEntryArchives'; +import getSchemaForName from '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName'; + +const MOCK_DATA = [{ Message__c: 'Some message', TransactionId__c: 'ABC-123', TransactionEntryNumber__c: 1 }]; +const MOCK_SCHEMA = require('./data/LoggerSObjectMetadata.getSchemaForName.json'); + +const SHOW_TOAST_EVENT_NAME = 'lightning__showtoast'; +const SHOW_TOAST_EVENT_HANDLER = jest.fn(); +jest.mock( + '@salesforce/apex/LogEntryArchiveController.getLogEntryArchives', + () => { + return { + default: jest.fn() + }; + }, + { virtual: true } +); +jest.mock( + '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName', + () => { + return { + default: jest.fn() + }; + }, + { virtual: true } +); + +async function initializeElement() { + const element = createElement('c-log-entry-archives', { + is: LogEntryArchives + }); + element.addEventListener(SHOW_TOAST_EVENT_NAME, SHOW_TOAST_EVENT_HANDLER); + document.body.appendChild(element); + + await Promise.resolve('resolves connectedCallback()'); + await Promise.resolve('resolves getSchemaForName()'); + await Promise.resolve('resolves getLogEntryArchives()'); + await Promise.resolve('resolves getLogEntryArchives().then()'); + await Promise.resolve('resolves lightning-datatable re-render'); + + return element; +} + +describe('c-log-entry-archives', () => { + afterEach(() => { + while (document.body.firstChild) { + document.body.removeChild(document.body.firstChild); + } + jest.clearAllMocks(); + }); + + it('initializes component', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + getLogEntryArchives.mockResolvedValue(MOCK_DATA); + + const element = await initializeElement(); + + expect(document.title).toEqual('Log Entry Archives'); + const expectedTableData = JSON.parse(JSON.stringify(MOCK_DATA)); + expectedTableData.forEach(record => { + record.compositeId = record.TransactionId__c + record.TransactionEntryNumber__c; + }); + const table = element.shadowRoot.querySelector('lightning-datatable'); + expect(table).toBeTruthy(); + expect(table.data).toEqual(expectedTableData); + expect(getSchemaForName).toHaveBeenCalledTimes(1); + expect(getLogEntryArchives).toHaveBeenCalledTimes(1); + const expectedApexParameters = { + endDate: element.shadowRoot.querySelector('[data-id="endDate"]').value, + messageSearchTerm: element.shadowRoot.querySelector('[data-id="messageSearch"]').value, + minimumLoggingLevelOrdinal: Number(element.shadowRoot.querySelector('[data-id="loggingLevelFilter"]').value), + rowLimit: Number(element.shadowRoot.querySelector('[data-id="rowLimitFilter"]').value), + startDate: element.shadowRoot.querySelector('[data-id="startDate"]').value + }; + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0]).toEqual(expectedApexParameters); + }); + + it('shows toast message for apex controller error', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + const mockApexError = new Error('It broke!'); + getLogEntryArchives.mockRejectedValue(mockApexError); + + const element = await initializeElement(); + + const table = element.shadowRoot.querySelector('lightning-datatable'); + expect(table).toBeTruthy(); + expect(table.data.length).toEqual(0); + expect(getSchemaForName).toHaveBeenCalledTimes(1); + expect(getLogEntryArchives).toHaveBeenCalledTimes(1); + const expectedApexParameters = { + endDate: element.shadowRoot.querySelector('[data-id="endDate"]').value, + messageSearchTerm: element.shadowRoot.querySelector('[data-id="messageSearch"]').value, + minimumLoggingLevelOrdinal: Number(element.shadowRoot.querySelector('[data-id="loggingLevelFilter"]').value), + rowLimit: Number(element.shadowRoot.querySelector('[data-id="rowLimitFilter"]').value), + startDate: element.shadowRoot.querySelector('[data-id="startDate"]').value + }; + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0]).toEqual(expectedApexParameters); + expect(SHOW_TOAST_EVENT_HANDLER).toBeCalledTimes(1); + }); + + it('requeries when start date changes', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + getLogEntryArchives.mockResolvedValue(MOCK_DATA); + const element = await initializeElement(); + const threeDaysAgo = new Date(new Date().getTime() + -3 * 24 * 60 * 60 * 1000).toISOString(); + expect(getLogEntryArchives).toHaveBeenCalledTimes(1); + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].startDate).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].startDate).not.toEqual(threeDaysAgo); + + const startDateInput = element.shadowRoot.querySelector('[data-id="startDate"]'); + startDateInput.dispatchEvent( + new CustomEvent('change', { + detail: { + value: threeDaysAgo + } + }) + ); + + await Promise.resolve('resolves getLogEntryArchives()'); + await Promise.resolve('resolves getLogEntryArchives().then()'); + await Promise.resolve('resolves lightning-datatable re-render'); + expect(getLogEntryArchives.mock.calls[1][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[1][0].startDate).toEqual(threeDaysAgo); + }); + + it('requeries when end date changes', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + getLogEntryArchives.mockResolvedValue(MOCK_DATA); + const element = await initializeElement(); + const threeDaysAgo = new Date(new Date().getTime() + -3 * 24 * 60 * 60 * 1000).toISOString(); + expect(getLogEntryArchives).toHaveBeenCalledTimes(1); + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].endDate).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].endDate).not.toEqual(threeDaysAgo); + + const endDateInput = element.shadowRoot.querySelector('[data-id="endDate"]'); + endDateInput.dispatchEvent( + new CustomEvent('change', { + detail: { + value: threeDaysAgo + } + }) + ); + + await Promise.resolve('resolves getLogEntryArchives()'); + await Promise.resolve('resolves getLogEntryArchives().then()'); + await Promise.resolve('resolves lightning-datatable re-render'); + expect(getLogEntryArchives.mock.calls[1][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[1][0].endDate).toEqual(threeDaysAgo); + }); + + it('requeries when logging level changes', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + getLogEntryArchives.mockResolvedValue(MOCK_DATA); + const element = await initializeElement(); + const errorLoggingLevelOrdinal = '8'; + expect(getLogEntryArchives).toHaveBeenCalledTimes(1); + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].minimumLoggingLevelOrdinal).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].minimumLoggingLevelOrdinal).not.toEqual(Number(errorLoggingLevelOrdinal)); + + const minimumLoggingLevelOrdinalInput = element.shadowRoot.querySelector('[data-id="loggingLevelFilter"]'); + minimumLoggingLevelOrdinalInput.value = errorLoggingLevelOrdinal; + minimumLoggingLevelOrdinalInput.dispatchEvent( + new CustomEvent('change', { + target: { + value: errorLoggingLevelOrdinal + } + }) + ); + + await Promise.resolve('resolves getLogEntryArchives()'); + await Promise.resolve('resolves getLogEntryArchives().then()'); + await Promise.resolve('resolves lightning-datatable re-render'); + expect(getLogEntryArchives.mock.calls[1][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[1][0].minimumLoggingLevelOrdinal).toEqual(Number(errorLoggingLevelOrdinal)); + }); + + it('requeries when message search term changes', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + getLogEntryArchives.mockResolvedValue(MOCK_DATA); + const element = await initializeElement(); + const newMessageSearchTerm = 'hello, is this thing working??'; + expect(getLogEntryArchives).toHaveBeenCalledTimes(1); + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].messageSearchTerm).toBeUndefined(); + expect(getLogEntryArchives.mock.calls[0][0].messageSearchTerm).not.toEqual(newMessageSearchTerm); + + const messageSearchTermFilterInput = element.shadowRoot.querySelector('[data-id="messageSearch"]'); + messageSearchTermFilterInput.value = newMessageSearchTerm; + messageSearchTermFilterInput.dispatchEvent( + new CustomEvent('change', { + target: { + value: newMessageSearchTerm + } + }) + ); + + await Promise.resolve('resolves getLogEntryArchives()'); + await Promise.resolve('resolves getLogEntryArchives().then()'); + await Promise.resolve('resolves lightning-datatable re-render'); + expect(getLogEntryArchives.mock.calls[1][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[1][0].messageSearchTerm).toEqual(newMessageSearchTerm); + }); + + it('requeries when row limit changes', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + getLogEntryArchives.mockResolvedValue(MOCK_DATA); + const element = await initializeElement(); + const newRowLimit = '987654321'; + expect(getLogEntryArchives).toHaveBeenCalledTimes(1); + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].rowLimit).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0].rowLimit).not.toEqual(Number(newRowLimit)); + + const rowLimitFilterInput = element.shadowRoot.querySelector('[data-id="rowLimitFilter"]'); + rowLimitFilterInput.value = newRowLimit; + rowLimitFilterInput.dispatchEvent( + new CustomEvent('change', { + target: { + value: newRowLimit + } + }) + ); + + await Promise.resolve('resolves getLogEntryArchives()'); + await Promise.resolve('resolves getLogEntryArchives().then()'); + await Promise.resolve('resolves lightning-datatable re-render'); + expect(getLogEntryArchives.mock.calls[1][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[1][0].rowLimit).toEqual(Number(newRowLimit)); + }); + + it('requeries when refresh button is clicked', async () => { + getSchemaForName.mockResolvedValue(MOCK_SCHEMA); + getLogEntryArchives.mockResolvedValue(MOCK_DATA); + const element = await initializeElement(); + const expectedApexParameters = { + endDate: element.shadowRoot.querySelector('[data-id="endDate"]').value, + messageSearchTerm: element.shadowRoot.querySelector('[data-id="messageSearch"]').value, + minimumLoggingLevelOrdinal: Number(element.shadowRoot.querySelector('[data-id="loggingLevelFilter"]').value), + rowLimit: Number(element.shadowRoot.querySelector('[data-id="rowLimitFilter"]').value), + startDate: element.shadowRoot.querySelector('[data-id="startDate"]').value + }; + expect(getLogEntryArchives.mock.calls[0][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[0][0]).toEqual(expectedApexParameters); + + const refreshButton = element.shadowRoot.querySelector('[data-id="refresh-btn"]'); + refreshButton.click(); + + await Promise.resolve('resolves getLogEntryArchives()'); + await Promise.resolve('resolves getLogEntryArchives().then()'); + await Promise.resolve('resolves lightning-datatable re-render'); + expect(getLogEntryArchives.mock.calls[1][0]).toBeDefined(); + expect(getLogEntryArchives.mock.calls[1][0]).toEqual(expectedApexParameters); + }); +}); diff --git a/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.html b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.html new file mode 100644 index 000000000..a69c6694e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.html @@ -0,0 +1,88 @@ + + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js new file mode 100644 index 000000000..b03e435ed --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js @@ -0,0 +1,195 @@ +/************************************************************************************************* + * This file is part of the Nebula Logger project, released under the MIT License. * + * See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. * + ************************************************************************************************/ + +import { LightningElement } from 'lwc'; +import { ShowToastEvent } from 'lightning/platformShowToastEvent'; +import getSchemaForName from '@salesforce/apex/LoggerSObjectMetadata.getSchemaForName'; +import getLogEntryArchives from '@salesforce/apex/LogEntryArchiveController.getLogEntryArchives'; + +export default class LogEntryArchives extends LightningElement { + isLoading = false; + logEntryArchives = []; + startDate = new Date(new Date().getTime() + -60 * 24 * 60 * 60 * 1000).toISOString(); // 60 days ago + endDate = new Date().toISOString(); + minimumLoggingLevelOrdinal; + rowLimit; + _schema = {}; + _messageSearchTerm; + + async connectedCallback() { + this.minimumLoggingLevelOrdinal = this.loggingLevelOptions[this.loggingLevelOptions.length - 1].value; + this.rowLimit = this.rowLimitOptions[0].value; + await getSchemaForName({ sobjectApiName: 'LogEntryArchive__b' }).then(data => { + this._schema = data; + document.title = this._schema.labelPlural; + }); + this._loadColumns(); + await this._loadLogEntryArchives(); + } + + get title() { + return this.logEntryArchives.length + ' ' + this._schema.labelPlural; + } + + get loggingLevelOptions() { + return [ + { label: 'ERROR', value: '8' }, + { label: 'WARN', value: '7' }, + { label: 'INFO', value: '6' }, + { label: 'DEBUG', value: '5' }, + { label: 'FINE', value: '4' }, + { label: 'FINER', value: '3' }, + { label: 'FINEST', value: '2' } + ]; + } + + get rowLimitOptions() { + return [ + { label: '50 Records', value: '50' }, + { label: '100 Records', value: '100' } + ]; + } + + handleDateChange(event) { + const updatedProperty = event.target.dataset.id; + this[updatedProperty] = !event.detail.value ? '' : new Date(event.detail.value).toISOString(); + this._loadLogEntryArchives(); + } + + handleLoggingLevelFilterChange(event) { + const selectedLoggingLevelOrdinal = event.target.value; + this.minimumLoggingLevelOrdinal = selectedLoggingLevelOrdinal; + this._loadLogEntryArchives(); + } + + handleRowLimitFilterChange(event) { + const selectedRowLimit = event.target.value; + this.rowLimit = selectedRowLimit; + this._loadLogEntryArchives(); + } + + handleSearch(event) { + this._messageSearchTerm = event.target.value; + this._loadLogEntryArchives(); + } + + handleRefresh() { + const searchTerm = this.template.querySelector('lightning-input').value; + this._loadLogEntryArchives(searchTerm); + } + + // TODO Future release, add 'View' action with ability to view more fields for a `LogEntryArchive__b` record (similar to loggerSettings LWC) + // handleRowAction(event) { + // const actionName = event.detail.action.name; + // const row = event.detail.row; + // /* eslint-disable-next-line default-case */ + // switch (actionName) { + // case 'view': + // alert('TODO!'); + // // this._viewCurrentRecord(row); + // break; + // } + // } + + async _loadLogEntryArchives() { + let hasInvalidInputs = false; + this.template.querySelectorAll('lightning-input, lightning-comboxbox').forEach(input => { + if (input.reportValidity() === false) { + hasInvalidInputs = true; + } + }); + + if (hasInvalidInputs === true) { + return; + } + + this.isLoading = true; + this.logEntryArchives = []; + getLogEntryArchives({ + startDate: this.startDate, + endDate: this.endDate, + rowLimit: Number(this.rowLimit), + messageSearchTerm: this._messageSearchTerm, + minimumLoggingLevelOrdinal: Number(this.minimumLoggingLevelOrdinal) + }) + .then(results => { + this.logEntryArchives = JSON.parse(JSON.stringify(results)); + + this.logEntryArchives.forEach(archive => { + archive.compositeId = archive.TransactionId__c + archive.TransactionEntryNumber__c; + }); + this.isLoading = false; + }) + .catch(this._handleError); + } + + _loadColumns() { + this.columns = []; + + const tableColumnNames = [ + 'LoggedByUsername__c', + 'TransactionId__c', + 'TransactionEntryNumber__c', + 'LoggingLevel__c', + 'OriginType__c', + 'OriginLocation__c', + 'Message__c', + 'Timestamp__c' + ]; + for (let i = 0; i < tableColumnNames.length; i++) { + const field = this._schema.fields[tableColumnNames[i]]; + const column = { + fieldName: field.localApiName, + label: field.label, + type: field.type.toLowerCase() + }; + if (column.type === 'datetime') { + column.type = 'date'; + // FIXME and make dynamic based on user prefences for datetimes + column.typeAttributes = { + month: '2-digit', + day: '2-digit', + year: 'numeric', + hour: '2-digit', + minute: '2-digit', + second: '2-digit' + }; + } else if (column.type === 'string') { + column.type = 'text'; + } else if (column.type === 'textarea') { + column.wrapText = true; + } + this.columns.push(column); + } + + // TODO Finish in a future release + // Append the row-level actions + // let tableRowActions = [{ label: 'View', name: 'view' }]; + // this.columns.push({ + // type: 'action', + // typeAttributes: { + // menuAlignment: 'auto' + // rowActions: tableRowActions, + // } + // }); + } + + _handleError = error => { + const errorMessage = error.body ? error.body.message : 'error.message'; + const errorStackTrace = error.body ? error.body.stackTrace : ''; + const errorExceptionType = error.body ? error.body.exceptionType + ' in ' : ''; + /* eslint-disable-next-line no-console */ + console.error({ error }); + this.dispatchEvent( + new ShowToastEvent({ + message: errorExceptionType + errorStackTrace, + mode: 'sticky', + title: errorMessage, + variant: 'error' + }) + ); + this.isLoading = false; + }; +} diff --git a/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js-meta.xml new file mode 100644 index 000000000..250bdecf9 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/lwc/logEntryArchives/logEntryArchives.js-meta.xml @@ -0,0 +1,9 @@ + + + 54.0 + false + Log Entry Archives + + lightning__Tab + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/LogEntryArchive__b.object-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/LogEntryArchive__b.object-meta.xml new file mode 100644 index 000000000..01a7b7f72 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/LogEntryArchive__b.object-meta.xml @@ -0,0 +1,8 @@ + + + Deployed + Big Object representation of Logger data, used as an alternative to the platform event LogEntryEvent__e, as well as a way to archive Logger data stored in Log__c, LogEntry__, and LogEntryTag__c + + Log Entry Archives + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseNumber__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseNumber__c.field-meta.xml new file mode 100644 index 000000000..86c92f2ad --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseNumber__c.field-meta.xml @@ -0,0 +1,14 @@ + + + ApiReleaseNumber__c + Active + The release number for the org's instance - determined by making a callout to status.salesforce.com + false + The release number for the org's instance - determined by making a callout to status.salesforce.com + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseVersion__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseVersion__c.field-meta.xml new file mode 100644 index 000000000..3d5985c64 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiReleaseVersion__c.field-meta.xml @@ -0,0 +1,14 @@ + + + ApiReleaseVersion__c + Active + The release version for the org's instance - determined by making a callout to status.salesforce.com + false + The release version for the org's instance - determined by making a callout to status.salesforce.com + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiVersion__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiVersion__c.field-meta.xml new file mode 100644 index 000000000..6d5b6fc42 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ApiVersion__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ApiVersion__c + false + + 5 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchiveRetentionDate__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchiveRetentionDate__c.field-meta.xml new file mode 100644 index 000000000..0f041325e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchiveRetentionDate__c.field-meta.xml @@ -0,0 +1,11 @@ + + + ArchiveRetentionDate__c + Active + Reserved for future use + false + + false + Confidential + DateTime + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedById__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedById__c.field-meta.xml new file mode 100644 index 000000000..bf45424fe --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedById__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ArchivedById__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedByUsername__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedByUsername__c.field-meta.xml new file mode 100644 index 000000000..c615e70a0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedByUsername__c.field-meta.xml @@ -0,0 +1,12 @@ + + + ArchivedByUsername__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedDate__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedDate__c.field-meta.xml new file mode 100644 index 000000000..fe628a97e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ArchivedDate__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ArchivedDate__c + Active + false + + false + Confidential + DateTime + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedById__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedById__c.field-meta.xml new file mode 100644 index 000000000..ff03d352b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedById__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ClosedById__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedByUsername__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedByUsername__c.field-meta.xml new file mode 100644 index 000000000..a45874fe4 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedByUsername__c.field-meta.xml @@ -0,0 +1,12 @@ + + + ClosedByUsername__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedDate__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedDate__c.field-meta.xml new file mode 100644 index 000000000..7dc1533db --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ClosedDate__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ClosedDate__c + Active + false + + false + Confidential + DateTime + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Comments__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Comments__c.field-meta.xml new file mode 100644 index 000000000..71b45059f --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Comments__c.field-meta.xml @@ -0,0 +1,11 @@ + + + Comments__c + Active + false + + 4000 + Confidential + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ComponentType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ComponentType__c.field-meta.xml new file mode 100644 index 000000000..8c1d1acf3 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ComponentType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ComponentType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionSize__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionSize__c.field-meta.xml new file mode 100644 index 000000000..bbe1159fb --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionSize__c.field-meta.xml @@ -0,0 +1,13 @@ + + + DatabaseResultCollectionSize__c + The number of items contained in the collection of database results + false + The number of items contained in the collection of database results + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionType__c.field-meta.xml new file mode 100644 index 000000000..810981951 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultCollectionType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + DatabaseResultCollectionType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultJson__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultJson__c.field-meta.xml new file mode 100644 index 000000000..0df56c9cc --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultJson__c.field-meta.xml @@ -0,0 +1,9 @@ + + + DatabaseResultJson__c + false + + 131072 + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultType__c.field-meta.xml new file mode 100644 index 000000000..727a0c496 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/DatabaseResultType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + DatabaseResultType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EpochTimestamp__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EpochTimestamp__c.field-meta.xml new file mode 100644 index 000000000..921292c5e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EpochTimestamp__c.field-meta.xml @@ -0,0 +1,12 @@ + + + EpochTimestamp__c + Timestamp in milliseconds elapsed since 1 January 1970 of the log event + false + + 18 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EventUuid__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EventUuid__c.field-meta.xml new file mode 100644 index 000000000..81ade15d2 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/EventUuid__c.field-meta.xml @@ -0,0 +1,13 @@ + + + EventUuid__c + false + The UUID of the LogEntryEvent__e platform event that created this LogEntry__c record. + +For more details, refer to https://developer.salesforce.com/docs/atlas.en-us.platform_events.meta/platform_events/platform_events_event_uuid.htm + + 36 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionMessage__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionMessage__c.field-meta.xml new file mode 100644 index 000000000..33edbb88f --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionMessage__c.field-meta.xml @@ -0,0 +1,9 @@ + + + ExceptionMessage__c + false + + 131072 + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionStackTrace__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionStackTrace__c.field-meta.xml new file mode 100644 index 000000000..a6a1ea40f --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionStackTrace__c.field-meta.xml @@ -0,0 +1,9 @@ + + + ExceptionStackTrace__c + false + + 131072 + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionType__c.field-meta.xml new file mode 100644 index 000000000..b9c80718f --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ExceptionType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ExceptionType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBodyMasked__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBodyMasked__c.field-meta.xml new file mode 100644 index 000000000..94856d295 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBodyMasked__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpRequestBodyMasked__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBody__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBody__c.field-meta.xml new file mode 100644 index 000000000..50414a97a --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestBody__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpRequestBody__c + false + + 131072 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestCompressed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestCompressed__c.field-meta.xml new file mode 100644 index 000000000..a104eac3a --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestCompressed__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpRequestCompressed__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestEndpoint__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestEndpoint__c.field-meta.xml new file mode 100644 index 000000000..f50bbbd77 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestEndpoint__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpRequestEndpoint__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestMethod__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestMethod__c.field-meta.xml new file mode 100644 index 000000000..2523f3b51 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpRequestMethod__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpRequestMethod__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBodyMasked__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBodyMasked__c.field-meta.xml new file mode 100644 index 000000000..fd3396920 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBodyMasked__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpResponseBodyMasked__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBody__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBody__c.field-meta.xml new file mode 100644 index 000000000..b0f892ac1 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseBody__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseBody__c + false + + 131072 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseHeaderKeys__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseHeaderKeys__c.field-meta.xml new file mode 100644 index 000000000..d5ca1263f --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseHeaderKeys__c.field-meta.xml @@ -0,0 +1,11 @@ + + + HttpResponseHeaderKeys__c + false + + 1000 + false + false + LongTextArea + 5 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatusCode__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatusCode__c.field-meta.xml new file mode 100644 index 000000000..d3f8431d7 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatusCode__c.field-meta.xml @@ -0,0 +1,13 @@ + + + HttpResponseStatusCode__c + false + + 10 + false + 0 + false + false + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatus__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatus__c.field-meta.xml new file mode 100644 index 000000000..a12ff395e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/HttpResponseStatus__c.field-meta.xml @@ -0,0 +1,12 @@ + + + HttpResponseStatus__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsClosed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsClosed__c.field-meta.xml new file mode 100644 index 000000000..90ebebcb7 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsClosed__c.field-meta.xml @@ -0,0 +1,12 @@ + + + IsClosed__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsResolved__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsResolved__c.field-meta.xml new file mode 100644 index 000000000..9d0f2903a --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/IsResolved__c.field-meta.xml @@ -0,0 +1,12 @@ + + + IsResolved__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Issue__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Issue__c.field-meta.xml new file mode 100644 index 000000000..3638f0ea5 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Issue__c.field-meta.xml @@ -0,0 +1,12 @@ + + + Issue__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesMax__c.field-meta.xml new file mode 100644 index 000000000..1a5aa22ac --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsAggregateQueriesMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesUsed__c.field-meta.xml new file mode 100644 index 000000000..4d1708dbe --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueriesUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsAggregateQueriesUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueryMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueryMax__c.field-meta.xml new file mode 100644 index 000000000..395293700 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAggregateQueryMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsAggregateQueryMax__c + false + + 18 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsMax__c.field-meta.xml new file mode 100644 index 000000000..d8146ed36 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsAsyncCallsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsUsed__c.field-meta.xml new file mode 100644 index 000000000..d3deaa52b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsAsyncCallsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsAsyncCallsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsMax__c.field-meta.xml new file mode 100644 index 000000000..206538c47 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsCalloutsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsUsed__c.field-meta.xml new file mode 100644 index 000000000..a608f498b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCalloutsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsCalloutsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeMax__c.field-meta.xml new file mode 100644 index 000000000..76f0eb2e8 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsCpuTimeMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeUsed__c.field-meta.xml new file mode 100644 index 000000000..6aca53f40 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsCpuTimeUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsCpuTimeUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsMax__c.field-meta.xml new file mode 100644 index 000000000..6f048594d --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsDmlRowsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsUsed__c.field-meta.xml new file mode 100644 index 000000000..0e7a9f58b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlRowsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsDmlRowsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsMax__c.field-meta.xml new file mode 100644 index 000000000..8faebd4c7 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsDmlStatementsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsUsed__c.field-meta.xml new file mode 100644 index 000000000..c1286ddf2 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsDmlStatementsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsDmlStatementsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsMax__c.field-meta.xml new file mode 100644 index 000000000..e1726086c --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsEmailInvocationsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsUsed__c.field-meta.xml new file mode 100644 index 000000000..559264e47 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsEmailInvocationsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsEmailInvocationsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsMax__c.field-meta.xml new file mode 100644 index 000000000..a4563ec30 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsFutureCallsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsUsed__c.field-meta.xml new file mode 100644 index 000000000..132bf6a7a --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsFutureCallsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsFutureCallsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeMax__c.field-meta.xml new file mode 100644 index 000000000..68f59cd51 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsHeapSizeMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeUsed__c.field-meta.xml new file mode 100644 index 000000000..3a1e4aaba --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsHeapSizeUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsHeapSizeUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsMax__c.field-meta.xml new file mode 100644 index 000000000..8690db56c --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsMobilePushApexCallsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsUsed__c.field-meta.xml new file mode 100644 index 000000000..dad758713 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsMobilePushApexCallsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsMobilePushApexCallsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsMax__c.field-meta.xml new file mode 100644 index 000000000..ee331234c --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsPublishImmediateDmlStatementsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsUsed__c.field-meta.xml new file mode 100644 index 000000000..62896e7ba --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsPublishImmediateDmlStatementsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsPublishImmediateDmlStatementsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsMax__c.field-meta.xml new file mode 100644 index 000000000..6a88164c6 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsQueueableJobsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsUsed__c.field-meta.xml new file mode 100644 index 000000000..bdbe9d240 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsQueueableJobsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsQueueableJobsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesMax__c.field-meta.xml new file mode 100644 index 000000000..cb1e6f3a5 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoqlQueriesMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesUsed__c.field-meta.xml new file mode 100644 index 000000000..49ca8daba --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueriesUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoqlQueriesUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsMax__c.field-meta.xml new file mode 100644 index 000000000..ba3c7f87e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoqlQueryLocatorRowsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsUsed__c.field-meta.xml new file mode 100644 index 000000000..f714b5a4b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryLocatorRowsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoqlQueryLocatorRowsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsMax__c.field-meta.xml new file mode 100644 index 000000000..ca9fb4a68 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoqlQueryRowsMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsUsed__c.field-meta.xml new file mode 100644 index 000000000..d787f60ba --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoqlQueryRowsUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoqlQueryRowsUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesMax__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesMax__c.field-meta.xml new file mode 100644 index 000000000..3be43682e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesMax__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoslSearchesMax__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesUsed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesUsed__c.field-meta.xml new file mode 100644 index 000000000..ef0b5b77d --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LimitsSoslSearchesUsed__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LimitsSoslSearchesUsed__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Locale__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Locale__c.field-meta.xml new file mode 100644 index 000000000..cc19b34b3 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Locale__c.field-meta.xml @@ -0,0 +1,10 @@ + + + Locale__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogEntryName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogEntryName__c.field-meta.xml new file mode 100644 index 000000000..3906f4524 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogEntryName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LogEntryName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogName__c.field-meta.xml new file mode 100644 index 000000000..2b95ce579 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LogName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogPurgeAction__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogPurgeAction__c.field-meta.xml new file mode 100644 index 000000000..c13fea9c8 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogPurgeAction__c.field-meta.xml @@ -0,0 +1,12 @@ + + + LogPurgeAction__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogRetentionDate__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogRetentionDate__c.field-meta.xml new file mode 100644 index 000000000..60f2877fa --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogRetentionDate__c.field-meta.xml @@ -0,0 +1,13 @@ + + + LogRetentionDate__c + Active + The date that this log can be automatically deleted by the batch job LogBatchPurger. + +It defaults to 2 weeks after creation (configurable in Logger Settings), but the date can be set manually or via automation if certain logs need to be kept longer/indefinitely. + false + + false + Confidential + DateTime + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedById__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedById__c.field-meta.xml new file mode 100644 index 000000000..ff7bcee95 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedById__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoggedById__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedByUsername__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedByUsername__c.field-meta.xml new file mode 100644 index 000000000..3e15933a0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggedByUsername__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoggedByUsername__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggerVersionNumber__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggerVersionNumber__c.field-meta.xml new file mode 100644 index 000000000..489c64afe --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggerVersionNumber__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoggerVersionNumber__c + false + + 14 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevelOrdinal__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevelOrdinal__c.field-meta.xml new file mode 100644 index 000000000..2d1c17756 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevelOrdinal__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LoggingLevelOrdinal__c + false + + 10 + true + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevel__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevel__c.field-meta.xml new file mode 100644 index 000000000..873c4debc --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoggingLevel__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoggingLevel__c + false + + 10 + true + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginApplication__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginApplication__c.field-meta.xml new file mode 100644 index 000000000..f245f5a00 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginApplication__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoginApplication__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginBrowser__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginBrowser__c.field-meta.xml new file mode 100644 index 000000000..90377e9f8 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginBrowser__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoginBrowser__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginDomain__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginDomain__c.field-meta.xml new file mode 100644 index 000000000..18e98770f --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginDomain__c.field-meta.xml @@ -0,0 +1,11 @@ + + + LoginDomain__c + DeprecateCandidate + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginHistoryId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginHistoryId__c.field-meta.xml new file mode 100644 index 000000000..32682f462 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginHistoryId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoginHistoryId__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginPlatform__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginPlatform__c.field-meta.xml new file mode 100644 index 000000000..d4f74610b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginPlatform__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoginPlatform__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginType__c.field-meta.xml new file mode 100644 index 000000000..e403396d0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LoginType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LoginType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogoutUrl__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogoutUrl__c.field-meta.xml new file mode 100644 index 000000000..405dd0709 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/LogoutUrl__c.field-meta.xml @@ -0,0 +1,10 @@ + + + LogoutUrl__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageMasked__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageMasked__c.field-meta.xml new file mode 100644 index 000000000..7d0932a18 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageMasked__c.field-meta.xml @@ -0,0 +1,12 @@ + + + MessageMasked__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageTruncated__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageTruncated__c.field-meta.xml new file mode 100644 index 000000000..c75dd21aa --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/MessageTruncated__c.field-meta.xml @@ -0,0 +1,12 @@ + + + MessageTruncated__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Message__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Message__c.field-meta.xml new file mode 100644 index 000000000..8a9dee90b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Message__c.field-meta.xml @@ -0,0 +1,9 @@ + + + Message__c + false + + 131072 + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkId__c.field-meta.xml new file mode 100644 index 000000000..958540dab --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + NetworkId__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLoginUrl__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLoginUrl__c.field-meta.xml new file mode 100644 index 000000000..0cefd5afb --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLoginUrl__c.field-meta.xml @@ -0,0 +1,10 @@ + + + NetworkLoginUrl__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLogoutUrl__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLogoutUrl__c.field-meta.xml new file mode 100644 index 000000000..696e41da0 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkLogoutUrl__c.field-meta.xml @@ -0,0 +1,10 @@ + + + NetworkLogoutUrl__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkName__c.field-meta.xml new file mode 100644 index 000000000..026c6052e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkName__c.field-meta.xml @@ -0,0 +1,11 @@ + + + NetworkName__c + The name of the user's Community site (based on NetworkId). + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkSelfRegistrationUrl__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkSelfRegistrationUrl__c.field-meta.xml new file mode 100644 index 000000000..dc6ebd30e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkSelfRegistrationUrl__c.field-meta.xml @@ -0,0 +1,10 @@ + + + NetworkSelfRegistrationUrl__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkUrlPathPrefix__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkUrlPathPrefix__c.field-meta.xml new file mode 100644 index 000000000..1c2aa374e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/NetworkUrlPathPrefix__c.field-meta.xml @@ -0,0 +1,12 @@ + + + NetworkUrlPathPrefix__c + The UrlPathPrefix is a unique string at the end of the URL for this community. For example, in the community URL CommunitiesSubdomainName.force.com/customers, customers is the UrlPathPrefix. + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationDomainUrl__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationDomainUrl__c.field-meta.xml new file mode 100644 index 000000000..35f55c010 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationDomainUrl__c.field-meta.xml @@ -0,0 +1,11 @@ + + + OrganizationDomainUrl__c + The value returned from Url..getOrgDomainUrl() + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationEnvironmentType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationEnvironmentType__c.field-meta.xml new file mode 100644 index 000000000..119b07710 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationEnvironmentType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OrganizationEnvironmentType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationId__c.field-meta.xml new file mode 100644 index 000000000..edf569108 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OrganizationId__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceName__c.field-meta.xml new file mode 100644 index 000000000..8742bbd70 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OrganizationInstanceName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml new file mode 100644 index 000000000..ae9380599 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationInstanceReleaseCycle__c.field-meta.xml @@ -0,0 +1,12 @@ + + + OrganizationInstanceReleaseCycle__c + DeprecateCandidate + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationName__c.field-meta.xml new file mode 100644 index 000000000..dc5ee2401 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OrganizationName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationNamespacePrefix__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationNamespacePrefix__c.field-meta.xml new file mode 100644 index 000000000..9b3d26997 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationNamespacePrefix__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OrganizationNamespacePrefix__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationType__c.field-meta.xml new file mode 100644 index 000000000..3ec26cfd7 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OrganizationType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OrganizationType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginLocation__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginLocation__c.field-meta.xml new file mode 100644 index 000000000..eab85c939 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginLocation__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OriginLocation__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginType__c.field-meta.xml new file mode 100644 index 000000000..9577a3cbe --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/OriginType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + OriginType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ParentLogTransactionId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ParentLogTransactionId__c.field-meta.xml new file mode 100644 index 000000000..bbeb28415 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ParentLogTransactionId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ParentLogTransactionId__c + false + + 36 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Priority__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Priority__c.field-meta.xml new file mode 100644 index 000000000..ec6f948a2 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Priority__c.field-meta.xml @@ -0,0 +1,12 @@ + + + Priority__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileId__c.field-meta.xml new file mode 100644 index 000000000..1eeb511fe --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ProfileId__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileName__c.field-meta.xml new file mode 100644 index 000000000..e69ef87e7 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ProfileName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ProfileName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionSize__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionSize__c.field-meta.xml new file mode 100644 index 000000000..127d1c711 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionSize__c.field-meta.xml @@ -0,0 +1,13 @@ + + + RecordCollectionSize__c + The number of items contained in the collection of database results + false + The number of items contained in the collection of database results + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionType__c.field-meta.xml new file mode 100644 index 000000000..487f3cd80 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordCollectionType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + RecordCollectionType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordId__c.field-meta.xml new file mode 100644 index 000000000..9f2403a04 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordId__c.field-meta.xml @@ -0,0 +1,12 @@ + + + RecordId__c + The ID of the record associated with a particular log entry. This is used for addRecordDebugEntry & addRecordExceptionEntry methods + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJsonMasked__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJsonMasked__c.field-meta.xml new file mode 100644 index 000000000..28c37cf52 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJsonMasked__c.field-meta.xml @@ -0,0 +1,12 @@ + + + RecordJsonMasked__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJson__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJson__c.field-meta.xml new file mode 100644 index 000000000..4b4eaf367 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordJson__c.field-meta.xml @@ -0,0 +1,9 @@ + + + RecordJson__c + false + + 131072 + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordName__c.field-meta.xml new file mode 100644 index 000000000..224a54acc --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordName__c.field-meta.xml @@ -0,0 +1,12 @@ + + + RecordName__c + false + + 255 + false + false + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectClassification__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectClassification__c.field-meta.xml new file mode 100644 index 000000000..da3bf3ed4 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectClassification__c.field-meta.xml @@ -0,0 +1,10 @@ + + + RecordSObjectClassification__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectTypeNamespace__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectTypeNamespace__c.field-meta.xml new file mode 100644 index 000000000..e151071b5 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectTypeNamespace__c.field-meta.xml @@ -0,0 +1,10 @@ + + + RecordSObjectTypeNamespace__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectType__c.field-meta.xml new file mode 100644 index 000000000..3a13a7454 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/RecordSObjectType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + RecordSObjectType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Scenario__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Scenario__c.field-meta.xml new file mode 100644 index 000000000..828189a30 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Scenario__c.field-meta.xml @@ -0,0 +1,10 @@ + + + Scenario__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionId__c.field-meta.xml new file mode 100644 index 000000000..50230878b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + SessionId__c + false + + 120 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionSecurityLevel__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionSecurityLevel__c.field-meta.xml new file mode 100644 index 000000000..9eb785c97 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionSecurityLevel__c.field-meta.xml @@ -0,0 +1,10 @@ + + + SessionSecurityLevel__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionType__c.field-meta.xml new file mode 100644 index 000000000..eb26425b5 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SessionType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + SessionType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SourceIp__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SourceIp__c.field-meta.xml new file mode 100644 index 000000000..6c78057dd --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SourceIp__c.field-meta.xml @@ -0,0 +1,10 @@ + + + SourceIp__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/StackTrace__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/StackTrace__c.field-meta.xml new file mode 100644 index 000000000..23390dbba --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/StackTrace__c.field-meta.xml @@ -0,0 +1,9 @@ + + + StackTrace__c + false + + 131072 + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Status__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Status__c.field-meta.xml new file mode 100644 index 000000000..34d4c4f63 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Status__c.field-meta.xml @@ -0,0 +1,12 @@ + + + Status__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SystemMode__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SystemMode__c.field-meta.xml new file mode 100644 index 000000000..fd2d2f399 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/SystemMode__c.field-meta.xml @@ -0,0 +1,10 @@ + + + SystemMode__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Tags__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Tags__c.field-meta.xml new file mode 100644 index 000000000..d8fa36708 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Tags__c.field-meta.xml @@ -0,0 +1,9 @@ + + + Tags__c + false + + 131072 + LongTextArea + 8 + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ThemeDisplayed__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ThemeDisplayed__c.field-meta.xml new file mode 100644 index 000000000..495f0517b --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/ThemeDisplayed__c.field-meta.xml @@ -0,0 +1,10 @@ + + + ThemeDisplayed__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneId__c.field-meta.xml new file mode 100644 index 000000000..75d3c06f8 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + TimeZoneId__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneName__c.field-meta.xml new file mode 100644 index 000000000..93aa4401e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimeZoneName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + TimeZoneName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimestampString__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimestampString__c.field-meta.xml new file mode 100644 index 000000000..43313af95 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TimestampString__c.field-meta.xml @@ -0,0 +1,12 @@ + + + TimestampString__c + This field is used to circumvent a platform limitation - platform event datetimes are not accurately transferred to Apex triggers. This field stores a string version of the timestamp, which is then converted back to a datetime in Apex triggers. + false + + 40 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Timestamp__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Timestamp__c.field-meta.xml new file mode 100644 index 000000000..7832d4511 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/Timestamp__c.field-meta.xml @@ -0,0 +1,8 @@ + + + Timestamp__c + false + + true + DateTime + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionEntryNumber__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionEntryNumber__c.field-meta.xml new file mode 100644 index 000000000..840fd561d --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionEntryNumber__c.field-meta.xml @@ -0,0 +1,12 @@ + + + TransactionEntryNumber__c + The sequential number of this log entry within the transaction + false + + 10 + true + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionId__c.field-meta.xml new file mode 100644 index 000000000..ba8b343d6 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TransactionId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + TransactionId__c + false + + 36 + true + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerIsExecuting__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerIsExecuting__c.field-meta.xml new file mode 100644 index 000000000..5d645fd36 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerIsExecuting__c.field-meta.xml @@ -0,0 +1,12 @@ + + + TriggerIsExecuting__c + Active + false + + 255 + false + Confidential + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerOperationType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerOperationType__c.field-meta.xml new file mode 100644 index 000000000..d272d60d1 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerOperationType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + TriggerOperationType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerSObjectType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerSObjectType__c.field-meta.xml new file mode 100644 index 000000000..a4afe50cb --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/TriggerSObjectType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + TriggerSObjectType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseDefinitionKey__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseDefinitionKey__c.field-meta.xml new file mode 100644 index 000000000..7617b03e9 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseDefinitionKey__c.field-meta.xml @@ -0,0 +1,11 @@ + + + UserLicenseDefinitionKey__c + https://developer.salesforce.com/docs/atlas.en-us.object_reference.meta/object_reference/sforce_api_objects_userlicense.htm + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseId__c.field-meta.xml new file mode 100644 index 000000000..30f7a3593 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + UserLicenseId__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseName__c.field-meta.xml new file mode 100644 index 000000000..db1956cb3 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLicenseName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + UserLicenseName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevelOrdinal__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevelOrdinal__c.field-meta.xml new file mode 100644 index 000000000..6fc350112 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevelOrdinal__c.field-meta.xml @@ -0,0 +1,11 @@ + + + UserLoggingLevelOrdinal__c + false + + 10 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevel__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevel__c.field-meta.xml new file mode 100644 index 000000000..66e885964 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserLoggingLevel__c.field-meta.xml @@ -0,0 +1,10 @@ + + + UserLoggingLevel__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleId__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleId__c.field-meta.xml new file mode 100644 index 000000000..137a32882 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleId__c.field-meta.xml @@ -0,0 +1,10 @@ + + + UserRoleId__c + false + + 18 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleName__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleName__c.field-meta.xml new file mode 100644 index 000000000..913b5976e --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserRoleName__c.field-meta.xml @@ -0,0 +1,10 @@ + + + UserRoleName__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserType__c.field-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserType__c.field-meta.xml new file mode 100644 index 000000000..5b53cf39c --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/fields/UserType__c.field-meta.xml @@ -0,0 +1,10 @@ + + + UserType__c + false + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/indexes/LogEntryArchiveIndex.indexe-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/indexes/LogEntryArchiveIndex.indexe-meta.xml new file mode 100644 index 000000000..b8c0ff879 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/objects/LogEntryArchive__b/indexes/LogEntryArchiveIndex.indexe-meta.xml @@ -0,0 +1,17 @@ + + + LogEntryArchiveIndex + + Timestamp__c + DESC + + + TransactionId__c + ASC + + + TransactionEntryNumber__c + DESC + + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/permissionsets/LoggerLogEntryArchiveAdmin.permissionset-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/permissionsets/LoggerLogEntryArchiveAdmin.permissionset-meta.xml new file mode 100644 index 000000000..1cc2bb531 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/permissionsets/LoggerLogEntryArchiveAdmin.permissionset-meta.xml @@ -0,0 +1,674 @@ + + + Provides admin access to the BigObject LogEntryEventArchive__b, included with the BigObject plugin for Nebula Logger + + true + LogEntryArchive__b.ApiReleaseNumber__c + true + + + true + LogEntryArchive__b.ApiReleaseVersion__c + true + + + true + LogEntryArchive__b.ApiVersion__c + true + + + true + LogEntryArchive__b.ClosedById__c + true + + + true + LogEntryArchive__b.ClosedByUsername__c + true + + + true + LogEntryArchive__b.ClosedDate__c + true + + + true + LogEntryArchive__b.Comments__c + true + + + true + LogEntryArchive__b.ComponentType__c + true + + + true + LogEntryArchive__b.DatabaseResultCollectionSize__c + true + + + true + LogEntryArchive__b.DatabaseResultCollectionType__c + true + + + true + LogEntryArchive__b.DatabaseResultJson__c + true + + + true + LogEntryArchive__b.DatabaseResultType__c + true + + + true + LogEntryArchive__b.EpochTimestamp__c + true + + + true + LogEntryArchive__b.ExceptionMessage__c + true + + + true + LogEntryArchive__b.ExceptionStackTrace__c + true + + + true + LogEntryArchive__b.ExceptionType__c + true + + + false + LogEntryArchive__b.HttpRequestBody__c + true + + + false + LogEntryArchive__b.HttpRequestBodyMasked__c + true + + + false + LogEntryArchive__b.HttpRequestEndpoint__c + true + + + false + LogEntryArchive__b.HttpRequestMethod__c + true + + + false + LogEntryArchive__b.HttpResponseBody__c + true + + + false + LogEntryArchive__b.HttpResponseBodyMasked__c + true + + + false + LogEntryArchive__b.HttpResponseHeaderKeys__c + true + + + false + LogEntryArchive__b.HttpResponseStatusCode__c + true + + + false + LogEntryArchive__b.HttpResponseStatus__c + true + + + true + LogEntryArchive__b.IsClosed__c + true + + + true + LogEntryArchive__b.IsResolved__c + true + + + true + LogEntryArchive__b.Issue__c + true + + + true + LogEntryArchive__b.LimitsAggregateQueriesMax__c + true + + + true + LogEntryArchive__b.LimitsAggregateQueriesUsed__c + true + + + true + LogEntryArchive__b.LimitsAggregateQueryMax__c + true + + + true + LogEntryArchive__b.LimitsAsyncCallsMax__c + true + + + true + LogEntryArchive__b.LimitsAsyncCallsUsed__c + true + + + true + LogEntryArchive__b.LimitsCalloutsMax__c + true + + + true + LogEntryArchive__b.LimitsCalloutsUsed__c + true + + + true + LogEntryArchive__b.LimitsCpuTimeMax__c + true + + + true + LogEntryArchive__b.LimitsCpuTimeUsed__c + true + + + true + LogEntryArchive__b.LimitsDmlRowsMax__c + true + + + true + LogEntryArchive__b.LimitsDmlRowsUsed__c + true + + + true + LogEntryArchive__b.LimitsDmlStatementsMax__c + true + + + true + LogEntryArchive__b.LimitsDmlStatementsUsed__c + true + + + true + LogEntryArchive__b.LimitsEmailInvocationsMax__c + true + + + true + LogEntryArchive__b.LimitsEmailInvocationsUsed__c + true + + + true + LogEntryArchive__b.LimitsFutureCallsMax__c + true + + + true + LogEntryArchive__b.LimitsFutureCallsUsed__c + true + + + true + LogEntryArchive__b.LimitsHeapSizeMax__c + true + + + true + LogEntryArchive__b.LimitsHeapSizeUsed__c + true + + + true + LogEntryArchive__b.LimitsMobilePushApexCallsMax__c + true + + + true + LogEntryArchive__b.LimitsMobilePushApexCallsUsed__c + true + + + true + LogEntryArchive__b.LimitsPublishImmediateDmlStatementsMax__c + true + + + true + LogEntryArchive__b.LimitsPublishImmediateDmlStatementsUsed__c + true + + + true + LogEntryArchive__b.LimitsQueueableJobsMax__c + true + + + true + LogEntryArchive__b.LimitsQueueableJobsUsed__c + true + + + true + LogEntryArchive__b.LimitsSoqlQueriesMax__c + true + + + true + LogEntryArchive__b.LimitsSoqlQueriesUsed__c + true + + + true + LogEntryArchive__b.LimitsSoqlQueryLocatorRowsMax__c + true + + + true + LogEntryArchive__b.LimitsSoqlQueryLocatorRowsUsed__c + true + + + true + LogEntryArchive__b.LimitsSoqlQueryRowsMax__c + true + + + true + LogEntryArchive__b.LimitsSoqlQueryRowsUsed__c + true + + + true + LogEntryArchive__b.LimitsSoslSearchesMax__c + true + + + true + LogEntryArchive__b.LimitsSoslSearchesUsed__c + true + + + true + LogEntryArchive__b.Locale__c + true + + + true + LogEntryArchive__b.LogEntryName__c + true + + + true + LogEntryArchive__b.LogName__c + true + + + true + LogEntryArchive__b.LogPurgeAction__c + true + + + true + LogEntryArchive__b.LogRetentionDate__c + true + + + true + LogEntryArchive__b.LoggedById__c + true + + + true + LogEntryArchive__b.LoggedByUsername__c + true + + + true + LogEntryArchive__b.LoggerVersionNumber__c + true + + + true + LogEntryArchive__b.LoginApplication__c + true + + + true + LogEntryArchive__b.LoginBrowser__c + true + + + true + LogEntryArchive__b.LoginDomain__c + true + + + true + LogEntryArchive__b.LoginHistoryId__c + true + + + true + LogEntryArchive__b.LoginPlatform__c + true + + + true + LogEntryArchive__b.LoginType__c + true + + + true + LogEntryArchive__b.LogoutUrl__c + true + + + true + LogEntryArchive__b.MessageMasked__c + true + + + true + LogEntryArchive__b.MessageTruncated__c + true + + + true + LogEntryArchive__b.Message__c + true + + + true + LogEntryArchive__b.NetworkId__c + true + + + true + LogEntryArchive__b.NetworkLoginUrl__c + true + + + true + LogEntryArchive__b.NetworkLogoutUrl__c + true + + + true + LogEntryArchive__b.NetworkName__c + true + + + true + LogEntryArchive__b.NetworkSelfRegistrationUrl__c + true + + + true + LogEntryArchive__b.NetworkUrlPathPrefix__c + true + + + true + LogEntryArchive__b.OrganizationDomainUrl__c + true + + + true + LogEntryArchive__b.OrganizationEnvironmentType__c + true + + + true + LogEntryArchive__b.OrganizationId__c + true + + + true + LogEntryArchive__b.OrganizationInstanceName__c + true + + + true + LogEntryArchive__b.OrganizationInstanceReleaseCycle__c + true + + + true + LogEntryArchive__b.OrganizationName__c + true + + + true + LogEntryArchive__b.OrganizationNamespacePrefix__c + true + + + true + LogEntryArchive__b.OrganizationType__c + true + + + true + LogEntryArchive__b.OriginLocation__c + true + + + true + LogEntryArchive__b.OriginType__c + true + + + true + LogEntryArchive__b.ParentLogTransactionId__c + true + + + true + LogEntryArchive__b.Priority__c + true + + + true + LogEntryArchive__b.ProfileId__c + true + + + true + LogEntryArchive__b.ProfileName__c + true + + + true + LogEntryArchive__b.RecordCollectionSize__c + true + + + true + LogEntryArchive__b.RecordCollectionType__c + true + + + true + LogEntryArchive__b.RecordId__c + true + + + true + LogEntryArchive__b.RecordJsonMasked__c + true + + + true + LogEntryArchive__b.RecordJson__c + true + + + true + LogEntryArchive__b.RecordSObjectClassification__c + true + + + true + LogEntryArchive__b.RecordSObjectTypeNamespace__c + true + + + true + LogEntryArchive__b.RecordSObjectType__c + true + + + true + LogEntryArchive__b.Scenario__c + true + + + true + LogEntryArchive__b.SessionId__c + true + + + true + LogEntryArchive__b.SessionSecurityLevel__c + true + + + true + LogEntryArchive__b.SessionType__c + true + + + true + LogEntryArchive__b.SourceIp__c + true + + + true + LogEntryArchive__b.StackTrace__c + true + + + true + LogEntryArchive__b.Status__c + true + + + true + LogEntryArchive__b.SystemMode__c + true + + + true + LogEntryArchive__b.Tags__c + true + + + true + LogEntryArchive__b.ThemeDisplayed__c + true + + + true + LogEntryArchive__b.TimeZoneId__c + true + + + true + LogEntryArchive__b.TimeZoneName__c + true + + + true + LogEntryArchive__b.TimestampString__c + true + + + true + LogEntryArchive__b.TriggerIsExecuting__c + true + + + true + LogEntryArchive__b.TriggerOperationType__c + true + + + true + LogEntryArchive__b.TriggerSObjectType__c + true + + + true + LogEntryArchive__b.UserLicenseDefinitionKey__c + true + + + true + LogEntryArchive__b.UserLicenseId__c + true + + + true + LogEntryArchive__b.UserLicenseName__c + true + + + true + LogEntryArchive__b.UserLoggingLevelOrdinal__c + true + + + true + LogEntryArchive__b.UserLoggingLevel__c + true + + + true + LogEntryArchive__b.UserRoleId__c + true + + + true + LogEntryArchive__b.UserRoleName__c + true + + + true + LogEntryArchive__b.UserType__c + true + + false + + + true + true + false + true + false + LogEntryArchive__b + false + + + LogEntryArchives + Visible + + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/tabs/LogEntryArchives.tab-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/tabs/LogEntryArchives.tab-meta.xml new file mode 100644 index 000000000..713f40dd2 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/tabs/LogEntryArchives.tab-meta.xml @@ -0,0 +1,6 @@ + + + + logEntryArchives + Custom13: Box + diff --git a/nebula-logger/plugins/big-object-archiving/plugin/testSuites/LoggerLogEntryArchivePlugin.testSuite-meta.xml b/nebula-logger/plugins/big-object-archiving/plugin/testSuites/LoggerLogEntryArchivePlugin.testSuite-meta.xml new file mode 100644 index 000000000..c1f33ed36 --- /dev/null +++ b/nebula-logger/plugins/big-object-archiving/plugin/testSuites/LoggerLogEntryArchivePlugin.testSuite-meta.xml @@ -0,0 +1,6 @@ + + + LogEntryArchiveBuilder_Tests + LogEntryArchiveController_Tests + LogEntryArchivePlugin_Tests + diff --git a/nebula-logger/plugins/log-retention-rules/.images/example-log-retention-rule-with-conditions.png b/nebula-logger/plugins/log-retention-rules/.images/example-log-retention-rule-with-conditions.png new file mode 100644 index 0000000000000000000000000000000000000000..112a4e1009a859a5c003a57ce059ad9ce94529bf GIT binary patch literal 186103 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%0AboCdM;x%<_u~b{wdSEAfy9P?`PQNp?N5bJ--Bd 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

    {1=cGX(>jggsN#qly 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 NZz^jmRVi47{y$yZK8^qY literal 0 HcmV?d00001 diff --git a/nebula-logger/plugins/log-retention-rules/.images/example-log-retention-rules-list-view.png b/nebula-logger/plugins/log-retention-rules/.images/example-log-retention-rules-list-view.png new file mode 100644 index 0000000000000000000000000000000000000000..78835b50c218ba8920e9a4f8a26c341b9e8a4e7e GIT binary patch literal 161345 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

    m|@MWbz-mgn;PU+Zr_Kfi!a>%u}wARJ9T5QL0}v zSaJxei(%YY+2oXi~{etExXGAE~SR4f$(tu4=?IVE!E!#eiD`wj>f$Vk!s zO`BGKntHGWG0@C5mY@Ipydhj0f<66EcI7y%#V1a$sS9%op;j?b&fyS-Sc<#R2QI(H z{9?q_&}+7NwE70sbAxxZ@r0P$g=b_ye{1vjs`UNLL>UX2Oc9&pzV7XKJ2HcHL*}Kw z@rv@2616_6BlzZIn1h(cr2`vReRlLxd&RNP*~Lxns0RPm1_CRSiM+Cbk_Y2f1SG)WN4obX-1PUX@(Q3pgG;< zuICvr265rD%&)l?Zr7|v4p04Mh!z$y-dU#d$&i!0lIhu_wndmcAWRtxIU1`-lk~-; zt+}Q}Wx;DEUU?!i7UM;*nqgZggH`ILla->;mY_p7K65gAykUE)D1^yRhx?Gf2HxdE7V%S8?$V{rrbYO>e zHfMC(Y0c7z>6@`{6EzZr@-waJU=5Ztv)0`8gd_Fy0*%*AD|S29E_>%D0x}l5K`(lyg2td-sB(GgcZ zbZ#bQ2s-&nv_Ukd&Bj`3rc}|{07dW~VblO@utIh4(EZWrrG|Rz-^&*E6LB0FF%8G1Y zaJ2;!6*lNTGEkBJ*nH1MQ&5)f6wHrcyGQ&eo42R9z+6J`u*j}Jej)qJ*dSA#gr}+I zI}k5Fg;Vt;m;|d}Hb2m_qo*FV)5IxST*4SyTGgce36c=ddCy=~OZ~!X)vv|a;b3Sj zG>ApdRH!yDuri>+To49r0u;YEe1_9W&V)uZOV>n~`)Kz@+vub|P++66Ij#-27Ckui zC?!=mgosraj#p(jz1yw%%>J3orz{8|N$WSitar7%NhRs?%;QO5v(G1GqZtzf<_V-U z=U~i0DVrY#XOHRoRwx|sEo78DQm}<{cX)yj{kBDMBasRc!aGAzgzRWHt?7K&7F4xC zmO2_d+mCrmI{Bi=)aF{;e%L*-_N#+tNnS({YX)IC!=@1-4ZyH$-egcm+ z7kNR2*W(Mr;*#xV?GGkiNQ|?}BWAh0V%c})n z=Amy8FTfJu1#~3O(+UvQJt^9OS`2|yAjgn4{}uXM=c*$Qn%S)x%8W<>{76 zrl>?s)i5emXoFcK1w9SpnM^05dr>$;oAwujj3BcF3WG@4mudPoW_1Y)ocRb!yh2jM zU*tysHX6Ir@kG zhC9PO!xKQxV~N6wa+ArL69i)8m`{sq4EssnN(pPyNDWk>TDFDSF1xIX@}CLFnX5nX zyP*o2POK4hX0T+;U$f`)xJXS{bOJdE0a1id$B%YD&|Oz)o-xoyE3L9Sp4`Z_RnPZK2`D$)j8s z>|h8~Qq$HSksNMJ?eUcQZ)Ck?R9wN*h8x`7U4pw4+})kv?ykWtxVyU!?rs5s%iu6r z2=2ixz@78mb=S#R_xH@&GrPOGx~jYCc`Nd!sxbHg1oon)V*qAI&3SNS=HYUhaD%U+ zt=YCh)WdL(R+ihdG{@QrxhhR`uI)UlN(9YJp(}@>1DHYV1 z;^JQEduIc~XGey60gNNmoW%&*b_>LYZ?kGUg4hLOCBs^Yf(H@1=QWK|JR+TJ@pNp$ zV|TMmI;?=AM%popUAu}y6Aq0UDvpjI)XC4CkOFe9LDYfsnnSq}z^hmH6+cX&m#;L4; z^PuQa2@b^AT}iCp!7-%}Tv~88%h39zuGK1xb{0Q7Hj^<@M_%k&Pgd%`{G9J$($0X7 zhj=bMX#DNQ$nz;T*3VRG(WK#mTrXsbAd~-KZWj7U$Dq&%e(WF?% zy-(gWMBV5pL`d~|L&krUEn**p(SE`ioo>c7E7I_6bTO1XMi)zcz^`eplCQkGu~CpZvGFm z#WTriR#=&aikwf{uk{J|oEHYtP=^%@r!!Xu14Zz8x+Hf>RS0IR9C|X$0;VWXu$-g! zh!s$Rh{P(8tF%RltZM$+O0Kl7HdWGl_VU7@B-PS)a(FP z4ad40tr$S&4ZTTQ74jKb$pT4ZjGGlo_Aa8qB4|((VHr*HaW!pI1>M|pP-azBHKJuZ zZqBq8#>AmG4s<&!i) z+T+_W>{UbFn-0up-ELQQHzc`^B5h?Ui1(Dc3E9nWklJ*3?1v&C+?6lQ-ml$(uXhcD zwXyzaE{Lmk$pNLZ^s%P}BO#uz#LZDHvVpjp?1U;5C^V-pc%S%FMx~6)rQL+XQSW0& z%WLnp3G>47-BpG#?vbhQdq6>!LY1iYRTRd}k~6V%36;7%T@xrQjVb{`Ag`6!q)?@R(%=eeAt}#x(z}zG3@){}y; z@9apFg`#!(BN~u@Jews%(U5B;nxs&yYZF7kiYQhmRJL0jb%*1NNiVFMxTewltrf2p z&q=eK=R(Ra-SK;Od2z$B>&g+jCS3B?D5(WBd+`K5O7yE+SuIEDx(ojvOu+0HIZwD8 zjrvl+OR<>Iv6>({$i-``SAe7T(9=AyD$_+|EVMypSpo5F&aDlYZ$_NvG+B*Vu49Sp2=8kI4s7>7$NU=4GL-U;{7L1SauR|qR!qU3`>N3 z7UL2HJ^X%1r0yv~AYF5<;91-Jo%scn)pp;@8*TeETq9NZ-^oT2l0q|%i8RO9hj_z~ ze88btI1-e@$YzTh9!2eu)VJy%M>Vbtq3vRa3M-6uF?V#VV1oW6bEs2jZ7p~N>6B6| z9yFg`HM!}k?)3@csP`R438f^iwP9u+sf!w9r4C9^;LU7XJstrvy%E(XTt>E8Et#qs zqltq{rzHjuU2umi&1N7iWlyM+iDHiW^LFjM4jaM+8Au!Cd*m!~-|;;!3e!^z`gsnw zx=?-mZkc6g>Av?45o__!igqL2egfa$ceJ3W~DxJ z9Q=Mm^bS3sPuy_Kv2cyB7Qf`(zIjH5EoT^+jZ!N0MbNd`w zgBYf^|4i^$LY-rbEG~{rf6i>BvANa#NGV=nZMtcH8}5}5A6^{if-cOQ3fdh8ciP1} zQMG2*)y*3u7L~gO{`l>^;Ld%)!>P8M5&}ygo72H9hyu?-f5+!o(U21A` zL~Sj0KdHaFHosjO=rlmKeFTYUtqHsIxxIS>u+p5Z=AqRl)uT6P9|32IDE7HB@p#)h z{N}!zj3+~jl8fTf*oWiLv*w!RsC|v>IQaP4?f{#G3C56g3tX0wHD?nIi~;vc7&d(e zH7*02u)!meg+O!>!lDv@@zvTPTb#ARlucihl>o1iomQs2sPyh+kiHdh)=XPEz6PR6 z_I>0e{EtDBSBm|#4PwM6!Yof^slw%yVjT)ah6M9^QN!aMQfD_?Rr%weWtsbcjQPZ~WEK!zVCtYVg&L?+L$M1)mvNflBJSlnt$ZQ?cVzN+^b zwdcjqzh5v}m+IKD7L}5^e+EDg9SXD?j)Qp<@OmVtBXKV0KfdP>Pfr&epE04`Qzxpzn5{QaqDsrK}Xoq#Zn0QFXmjkCPiScez5(Ru{2 z8yQkv#WBE%zZ5L-+RMX%AzY#36pRnMnN}g*o~-X0S>DE`P2il2G0m!jytgX+GJ}E+utP~UNf5}YojZRGXa|3Gls>j+O ziw!G)Os^_d=2bSYNO-KKk&)E|Xv$0mtiS|{%uM=_UWhdk!dv}+FsW}&LY z2+h9khdq2**8a5Br&w_l;RQ?kuBSvM>Q`dNE{X)pGWtn6865TwiW&1CC8mF9pbC(rC+A3W286o{mTR7WRm zFHhqSMHqC<6Pt!%VRn`_mxaDz{3h_AT@A3TZ?W=|bdl1`)&|JeO@H-5WV|pW zmA$8v6d#*F{>_Ltg`+f%56#^A$BijlML$8Uj^)IWgzq6Tj&|&sv=$gKF2NCD+x2|W zFFq`K;~Qu%jgrDg{E<77QEaV|UJ*!@CuF*}pWKfRC4@i6=gd?O(W@bWQb%GYxj2rL?ll zc04X~4e>_vj9X9<8L>r&0NV1&(mwzYl5x2mD&HDfEZ9$zrC7cp?^oETv>cL8(&z8C zUb&^L%G&wpF`BZ{cfrL+FrUxM>%CY#wc}q{Yyk4Hi(tN^UiP9En`$KpDWy`|g>fRh zkq|5&w%{#=EoQot?3-|t>!fjWIh^xHY}`0M5L$bUm}O83@K-7t-2JJ;WvGXWb3w$! zy>1^%V^i6xyJc{4PY35Kq?E-bR-w{*2ue!iW6+VN1@j&3lS`(vgTv!iYY(`XyF_vW+pX^g^w+%j^kl(a(+4U#%JkLYbB>tZY*7gXG+yc`s?7|B8l{t z-^Xa!m4nHepB~j8fbckk6shH)q%J|AJX7&D+Neva`{Jhuv{56^!BSpIg}a&^=_FHZ zcr6L)65Eq46s<|)IQ8GZWb`t!76`dN1amR<`Dob~ww0<6=HHX-ob+ z$RaFeH`(spTCWLOD&O@|;)Za_y#1Mq4?*eY6v8c!oD*vS#uvJIyLq4nrN7x>&;gPo z?v2c&CmTTvVoGVMB-6QcrO){=`x1Rg?vu{a1gh$j`bGJ{(P)|FO1Q@a>X#0QZJuKo z?@xpWJ`#F9^bwy{%OHxds$7M^H^ZbIzObtGTMGl@2pzn`5@G%ehC~Fzg0nNsTR+QZ zL!5NB|E6qPo5L?YV2+wPAxam={a}|oTbrC_40_MSN_I17zvs&IF%faV$GQvKhCGIk zBlH9lrEY>T#o=MJ#+b+pPRGCpIJk5)qo!TOAt_bXALU%8s2Aj9^h%Q6!%oFK_be>h zORNt3zD?3YAwrO1+@|Xhf1T>|8>zpLST3Ne<)*XKw3=M4Bf#2RBfD?)O)w15{?~4d@MN}7c%sc5&kB}0$aXBPYz*YB+A>Yr- zDemxeUSPR6@qP`AhXTs8Skq9a%g`f6a56#F0t5wYYeiM{CW>%Jb>QYP|Bs$x45maQ zrzmDT#AMuZ4?KYEWyk_r3C52^$hJ&K0qmo=cYdmi$kQ-JTa8VK4TC*D;!@PU8H~(@ z1Zvuc<134mvfG=ji2^uU^&@WNi+(nZj?3}a)P|%WX3vM35`E6X0L3vt8Vc7l1Unl3 zkTd-BunEScAYX!YGLX%!95BYK`HzPa*86+l!ivZvPuBrW0o(2EHkT_A-?lWyq&k^! z6@P3JUCbSad))oeVB@$D*)$YFt3rC9dWluym6kk>ddeeaWNld}H9GLD>q_SvV| zxbJ=lOd;satYwP7@x@n;bldK3RLL%zq~;URkP9T(_ZxriaWdH)SjRtk6b5hR1toMxaNi zexZbVjI1%`@1_akUsn01e=t$k6qvP;yW12RXPO!x3r({ELyzRnJUy9G2ZQC}7Eo47 z8g_P~`k-5Me5Y$oc+l-h8RrLRq7 zPhew`pJB_aeP8xM-qHBZAt6yVR>}{sJ1Z>;|Fm4)SI$1=8-(I1vqyynW6Umcey7mNcC}%h;rHo(V$iKOTM-M3Hzg} z{NOlGvJGZo&TwvQl$z?K^NAbP^2VQ~$Y?Bf_?Q)`M3d%`uT-Cfg5(Hqe2!_mWHd%f zVd*%G=gYvj26~Udv+6`>nk8rgmt28YP^pifY8Av(W%~kXo^+8pgy8d!3@7ey;cqsi z_x12jGO?7@M~7CgQM}$1m%neT(zJaRS~>}z=}%fng@6f5Le;^xw397D88&Kd>ilSS zBuQIM0w5i>1vR5-w3pmV8onxcyXdKfMI>dhVA)`3o12$Lw>1pTk!L}iAF9i`I8_aZ zCeB533I(G@wFS8NxfEGy&q>sJgmqCbH9)ai7ki7JpC-Gw!VuG?P-&N73DEVs5sfRS zrH-o?iE5rEbA+_$_KF?!y^XK2z_ssgD+NV@nY-ic<-*g_s^V4a){DnmEh5?>4XXi3u8|VlR4?Y02Q1n)RdkfFC?3fKZ9A&mx^-!v98G2j zAP1i)L_Gdnev?nM+!xAF>>~E_y8biu6xm!(D5vDr z#BBZ(dX6g@3@yi<07*iG9rtK486>H<d=DlZB$=FN!+MdaxJ;J5?sW|a4uOD9B{ckzWI{4(}aw& z>H%;v6*p8a?oe`^I+qv4$}Awtv_ur{u;_y=Q0nig5Cn{21My~#p0uriVFELAc_yv> zI7JwtVcWjBRP6};Jw{pl>;yu*MinHNyERjE1|uETAv$wp3Z{xxWX4W_1VC&c@<+F* z+2$mA=MH}m1(7DdHljEkesGn?v|0s%)tWdUhC^*Hc1lj9TwH;s6w5Ro*l==awG=q4 zs;lJeQ9bhsNe^Q(%x|8x#MOMkvZy=ahQgvP%A*Gm`vj2O7u|9ZtI&})mQB@VCS*6# z{(^nlFmvIEE-|bsXZcZdA>|nykjqi#S<}8tZ%f*)Q0DdmBtAUx0LTwQq7>K^VP1-S_GCh{M_BlbcfC@D`sRJqzV2A z9TKKkkMovBM9l4#z^D$o(N?o>9GLuBG=F#>w(P_W{V7WCrD>{97Zl>0cp&3T*u30b z{x!1qbmr9}21FG#k>IIN&M4ERH?}KH=&K?}nY=vGZNgKET$^Y3Q$$xh%UhNd!n{c1 zWBNy5iFx|q+Y?K6HLE!h>n{4zMQA3sH!1He>k9r2gJeN$ls>-j2g=PIe8GbGedpT6y@S0GifaN{c_K6)Wsz^BN5X5o<97jZ&cof4n zmlpDwTEr3MR`3%M=)8J(uJu^;hfWH5BO zvPg;My1CX+@g*mBx78%Ri=&_X)Od6DzFQ!(&4y8z)Zt_-P4GgKF9;FNpIu1rMCFg+ z8IA80Zdn93KNA-DRZn4BJk35o-peY@N%@@O7onYw&l6q66w7^wB<1JsuHsDwt&^5A z!JJaU>A9VDy4Y(v&Zq$glbU@K^cjxg=naU=#rEiJwxOdm>zisF4cZ0({81LFl6DUc zWlT;@Bjo$1Nr~^F+n`Ml31f@Sr+8Y6+TLyFA zM7p4jxA0RNGq&=|BN4k2fnxVYY5bi!LL!P%&1DU&sigRe@5@6)lR#r&8>7UPIf(40 zYcih1Aj3%UNctV@ha^zdBtc;g%gCZ{3uW}==Ek#};#3j}^{3BcP8zRGPeM*y2tLEG zC9I1(_dLKsc5(9}8lkNaX9CK3B5k>pI9%CfB9MOk&^}z>omb^Go3L?AR6uXj86*}~ zezI@*k90%%6&r@AWbHAa3e}dy;978j1@xFynu63(p}UQ<#TrOb>qEW&Jdgtnd=gH> zyNH}gtVL%r4C4uAn`yW$#-41cFyhmHO3HPU!Vm}KeI|G@96u|5g26%A^zhc~1D4R# z29lRMir3c}7@unW_CePq<98e61hB+)=JfcUv`}*@HV4mzL$uVjjm0rQ4MXD*$56n3YWT8wW6mcWvFI4<{;Ff! z_*Qy6y&=$<4DM+a&2eeo=1pZYx7x z{J?h^t$JyAD&>ehub6%FK&ir1biRrSL=xGR31Kr1X-AfdwrqMFluj$JsS0z9qKI9c zn~gCxo>ZVtWM!elgJ8X!#gp#Kpn~j}t9NLoEq6+ei%KZbkhF(e)l70)u0-uMKx|Cn z!k14XbUH%}UGtaJZGvPX_vYQ4mu8tNyrBx-tvt;_yNx~6Mmaxd)VQ^FxcD!_X-QZd zXCOiuQxn+n%acn~NbT86Xu_pJb_cIy{0_#2E{3Ki0m@vVa2g8p!ErG(&#r}?*JWH5 zpNVkctR&9uShpGyL4D}ik*`+I8$aa0-zBCPST^(O98zlJvmbF=e=lPGl z$@mHow47z;gX8;#(qKP3SPQ`Lu!<)Mnshtb#+%vLz18p5{2qg|yjc zl8_@&r)D`TcaIn4`7y{eyfXHS#pcs@*y)2vJ7%Mv5( zW|TrMH|U)FNIF&Hbpe}~PnkBw@0#79oZ3l_8Az4J4P34n@a{P=-tExq=_0;X#Age> zbI;5v8aweg&W^rD-zby?9DST}<<>P#CPGL8eaVE@@j2nRP>7zv`RIKQG83|~tCGX^ zp_2_74D@Z6E28eT7w4vCUeOuPE1~c*zql=kdGO7X;nqqe+|n|+GmlWNMPhdzfwlCt z8cl%2uG+~W`s_|<>I#wmF!O)W=D(5NG*Hmp$S+U6e zEHgBTql?%rGkzo;Hs3Q-l}w(l494|nL~V8^g*}A%0%EH|8a7=bZcJyz1LFnt`Y z!kFV#k1nWp)UUQCrUM&)s;W62%U?bRg4*@2bvE}qG2W1#U7Wg}{m0kh(jSth7Dhni zoZi?Z-r!i9+^1f2PW+Z!y$%h~3u|tR^|@a-w!6aGUEnO zpXi_zQNAi#OjXi@<@@3aNkdXCep$%NE{hEsm;ZHF6esunwN&I3UJCop zCka6a z(B)CJ2(jnWTumD`!rPg?X~zqR)+adBnh7^}HSv~C)P-q$5yik!9GW-Wvk#Tw(jrLR z74ln2gynK*<&|cYkai^#D_bh@ERuZ3$BeL2RgNd#&rX=wNr^zgWn~`d%%o7u)ZkOq zi`g_HjIfPM_3))40gARbux&r;n&1a}b#kHud1b9e55sQ6z7{*uK^&9^VuEki-054< z*(oFm|CmGqF=Xdp4DA%4McxKFCz2yH@P+70)v2vKNTV_edg4h1vZ$wRG3^h+e2eSM zAl1Sqp>WhAvfC5zA<}ItM^QX?(IsN9kpk>_s@l4fKmk>5-uPlw8q-|}+z@U**OV^o z#nR2I=QTjF5337lXPNllvGwO#==3J^fuBI7c6Osxis1`802$QAC?H=PiSD@p#Dyl_ zwTfevSp@!tl9OX>^_Yp}Wvn0{4Bru{eg;*(TB2c-2mXMEW%}@m<^U{_F^NUvno*z_ zrpu4fQxm1}*cl0PaLJx&#!Ec{x9+U=6=w{*IqSLm)#)+wEL(uPw1$?7akYs=7DnV> zI+#lx#+6v-5jL^n+6K##EylF$)!E6Pn#U5B(M%9cxQ$fnM(57?AVRR#BXV+y=^f5ra=dfty@(shZ zmm2G^F-agWXw3JB!|6Z3t{;wkSHDS!^GH2go~Dj9(DL%ea;}-pt{s)?5h0u&{SikPKr*F@5JAyVsZ7~K zC|oSz8VRXF_}rZMVr-i|0=54D85Lh-2x1P5djT1%q~dSgmp_#@WQ3s;6ch(4R%d)- zgFfxC2^|GFhEC_x)h19uSmEjgx+`eLE1d{cx&$b8OShR=h}wQEiC!`W3>yCm9$;vX z3bp70H|JIhZ~R@+4Q%V7EuL*kYO~DdPUYg=Pz%*^?zv$vO5r83-JW+EvoQy9TYY{Z zN|$rOx$_-eh1Kub>zY1>>K_5cLic{{anr}Q-WTK0gY;age8A_fDZ=N3dcIc=RtBKS zD$>|Dz8IdmhHDA>H*C9F|NTQY6_-&s)v|b!wT(Wi_#+Kyn$??`+D_XtwyUlq$UaG59I$S`|I>lM{8^P%ney4x|}xI>qv04FC-f4R34fOnYO^WI(C(J$ zJbOj9G|Mw?$%(1MN;c3#*nr}S-8Lm$p5}n>FQz0H>sh?(C&jH;9Kh;I|I>u>T}dnN zD&grj#6=yqai!xuvBJo8RK3viWc@cBy!oGH>NKn2;LDJzX7B-B#D}4rsPT8`7jig; zSlD()@5B`y0~w@9N^&nk*hPdlEVlr>oSB|c9gRO*YL-_-(MnmFZ0?`UH#W=BRJf+YGeyAm^ zV>dOwDR~vxDPOSVVxTN>UD%GftwnRY=I6`_zD9*|nqSzeO!X0}4p`u5bF*@rMV6nF zMR5)XVW#=BW$#vKepLjpYN3ZZRSFg79s0@7IAkQJ2@qZMqR(=?SNDE3FCw4ghd)rJ zz((*rVQO_bO>_0?XwuXwR8IMUbQNN+QjMkscv=C`jDm;$>o4OUq7o`_|l~G?ie?8iiTGvB=h&u zrnitI#~7`|d8{rTnJ$yeldG7gO#6u=lGhH)jleuhIz+J=$u$mDO@#8$20eeYGtEb| z=5tS4No50BGJ2_o`uCioOrBdWYqA}Ny@@a3$@u!oH|-t(m_~l~Kg^YE&PSKdZPN3a z-MPB+2Qc{|ZZ3H5gw~p&vA~G4Wy-Ja<+lZ?OFa2@O0ZDnD^Lch+VOdSyfVK2tmopM zlgP}iz6!qT#WTffqwM(stkukID07+oCbMK`%?1A?h8#q(VrGP#qckzvUue8-qru#q z3}_KQ67|Nbc5DI4c6lY3?t>RLHX5f(bMzjmiwZf5G;(@qD}iE@)JXub zJJ#ntA9A8<45E}SG^g9^xfR%wZ$YI>s4$V9U_g7o(_g^z=~KcS_TU z2peM&1_dIMF#6lW#1SAvqhb7*^XWxJEAGP4MEc7ZrOlSXb%E0(dQ64>S1s<4cAyJ~lLXFhCsQ|g$K+h!Zbo&OxB8@4G<#4>y+i5F41+WB#fDmd5S*xXh{^ccH5j_Ov2rVJWQibDR?;x!d#&xx7 zAZ*k99Uj0$otpt7ao7O7eoNdq*{=St_zJMHaXF`|i!0)O(ZnXC=9C;!oc+glVR&m{ z$r|apc$LH}i?-i3vwtMK+g}8T61s^aQRrCca^vg~*Q?+xG65tI6{hwnQ%{%NU6*s> zmPKBK`nYIYG47RLtsi3a_8_}Y)a$D54Ez{bkdopCU@R#(6*1D@OptT#FmhO|&nii5 zmKX*eFCMb^w<-dYNAHqQl~*-dRQ7OYHZze*U4T-X+f_)nEW|)u+f*Ar3B&9mO_{J~ zdQod(Z+5^y7v(B?v0zdKq?49JU+zgrE)3NwgV#?$~0&apA6VxHhng)-68E{$k?YRc6fq{@t)tAC=@5LU`D8)CfSs$pG^|Q(y zWB-xM`9co7KMwYg;PhAh{n&X|Ujzt=dk=sPa)%>CeThR!4K>OfE5BPPS)?gz@%kT4 zCvsxl*=!;1{I;N*I63h78~7$yZf}Oa#}9L~2j`)`(Lo+upafuD3=}YrJx| zOdIa5yYlZbKFYdPM;dSmFY(qLR#==#u)jmWk+^(#yOC*m90IW{v`)w)0nr{{$TUv2 zVtWtoa;_#Tg$Lio&5GDRzwkW_qGM{;ZLRb=kkf#)V^Iz3jzX>Jaww`JURL@dopyIdc4dx&k1b(?c8`L}SY>=qrh{W=yg0&#xIP4}`nhw7* zhi-3-3CjHYkS||=1Lj8!54!~Wmtdui;}(|;8#z}%oKC;{ z`~9QWsC_|KYO|Zmn2p2@S)G0`>;0N%($4sPs=% z9#J(6v8JM0#Er-M$M+rgxd88@8O9!_p2n8>;?%zu4JJKJC3 zxs|@jNYwL_+fI?^iuanU+>+}wuc_bKlVsWuO}MgLVxi<7%x~2a!f1U@=Wbuc1}X)Q z&Her{KWI}Vv&=fb448nzpq-}^&`XAH^SwY=xJys%Y4f1A&~V9rPP*)EZK7$_sgir0 zK%_Ps!Ur1xWGioDfZfU?$NeCA?cXp|ENCy0DpmjJ#{Z|rm zf_0UDh)1R*%G3i6&~(?6rsBjvg5RG)?9lGmMufX z0vKK%>4q%@v(5azDVHrwct~S|Q$G)87Ek%6sIDUX^yLuOjZ@It9%`VL#T-R=SlG5-W0z+FW+hl4IPLTs@*4Tf3@{o zlipy|LMxO4o3{9B3D@-Z{#ndF-F{loTa!7R;W7&UIRly2CEKn(<#Qqc3i5prtsNXhgLsuRk3OCMNf2PoTQvW;u z9(bE9HT8TCN{$Kv3V9FO%PVz*qpqL+^ZLNHZ<{Q_{?>!y5hm z;}CmBk+seLyw>CH8R&3S7816HV+`MAaQn>bx$iQV--KKTwU^-QK;Tw&df~6y1{krY z5VOD_u@{?6#%yK3EvI*8*l6Yh@FlrRb`o#VzgkJ%1P?+Q@f>ild3{bgz#{~>UMV`%{z~*H#eI3dqF@O5A`-LpZ~ND+?Qvef}))kpY7m4 z_xU0F`tIN5FsvYB{|KLLPL<)ngGrUvC89o6r|(O)P{Vw9tYBhZ(I0{Z%&Ydn<-SC$&|^sln{J*yKcB$n!pcS@~U z1LlRd4O;&cemtsj$T6&7ye0_lCrO|Y1RM{B==%lqI}g-_yy-i$fC3VQ7#JDXgCe8- zpT=i+MV?+J`R#wz_gu8?Bu1j$Rh$XG{o3+2_I+M#7Ls<*sefyedq#aFbKNiQd34m* zH@?06`%_7P#N$uGMRbqPa!o_6%a!XL0vHQG#3Sf;-o2+S1i9VcL*liM@p=WkWWp8y z>3rewCnSuQfw1^>F~oF@rb!q1sLg)$#&YaU6@Jq&ojCrUQD?;t@gKy4FT8 zTe0IE@$>O*8caZwMKyKC-=E_UmIwLko^yeJ4#0!Ch$n*G?;O+r`129#VOz}je_Rh~ znP+<~*SJMaZtna2EN{jX*eL#zdJi65em&f&*7@rB1wEd2z%goxM`jNJq_uohF~cu- z8ZaWs&;r;APLQN@FYqUhFrmBa5y`u)!BPQv?^pc@^X9j>D<0sxxYJM(AbD`<+iU3f zQHeOO)}I|T_R(*!w{!n}%DML?Z&CT>dDkyq_@#`K<#j4vWAw@*;O1%Ov9H7R+8cAX z&U?%LD!^gv0S6?79!TCgk^*-8F-eIJe%eG(KT=Tqc_McdytK8&Jt@)p-mlSAUZ;x; z=<@sj$CDw;-sepuo${s2S1d$Ja%9U!_+K{8;XL&CYJKxQHSSkxaA8Q-)n8Rz-Td9I zZW*ZK6P%E=igVW`R3Eo-apP-{s|WG6kCoVD^~_tDFZ{)M6}08Ov!_E_01frGxAm7) z`#Cxa()-npeQjOh-{I_1-`7@>y$#>HuLC4nxIR}O1y43p?>PPB4g^a)V<{(!mS2h* z9IJSX^{;)l0E!ii4`WYr&8%%Fk>hAR9{(7+gKz5tUw#5@>h(7!!q}g`Cw{y!qOP0g z?x7O;PDAv>;djrV>OXD=L_)w_2p=~Jc{h?pUiA&H1+;&hjFXHdf=q|1NO;;Ha~{1( zo36(FFF0FXFZ~?2UC5D%+OY0&j*Q=i5aXZLRbXvmi8n>~Z=&kz#)*(J0Saw+w*js9 zZj9Ww4a*zsSPGZmWxja0@kl9K>vxvCQ9p1@+2(pHm8qQD7uUabNY%XGuMiZpUEVS9 z65npaYhT#&{&=)y-TzU+ne_MlVx6}}a!%k1)Kh4j+LGBexu9j&adtbK>$NjwZ$V^R z?1!#UMtMzjCEZZhd{^nE z&^}sH1E5(bEF^Q@e1ZDrXTh}Ujgs75q1zn5is7UV<3WW9FdQI_2!ZOb$_+j5e`U+l zar_S1?Ydlf#$Y`Wjl74-J%cm1DfUWdb73Uel(Xn5bLe~}3NyF@hVkr-cSYtq1n8Pq=?vo zO#LD$LWY0`0O>oE)3CAnt09!Wm@V`c&hp!jCAz|P>C45Y&M`92DrMFSOTq>X8pTMD!peVD@RoNO7fbIB- zoB}Wss@`%^|I8}|PU*oy%#)UIjh1jDp`23X$hn#jjz<1Qx9KaAcoya&B5Q56hEJ!~ zEA29c>@+%i4zztvyCP13FS&I=Zo4q$ZF5CtiKQy)4QjXSfr&;Sio&fxhgjy;GvEbt zO>uEHi`}+Q-zexMsx)$g>#@H$lGH$uMQqJO6iB6UqREf-$2SdD4+`(%^Q+?Yg;)8K zieOsJKh_!ADCgQ`(EI2f(>ga{!J$Qp3Ip{rWpvV7zC8UC?Iv>It0S5Nc!v0~5$})J zu_6{W?0Y}n)$_cWn&mwF{&e?cw0T{4zchb38~9jhesR+D5Nke6xNL?NX2*&s=sL zV@+?&#{I@va~-^JdHs&yKIzrfsMOEwjY5KdECZ>Y#s0O>bD+Z1-R+$wNB$^Zx?E;@ zc$@xfBV(!Spj9eE(T~fXtB~8*tAWTcM`EzQ{MoQt8)CW>Ak(Y_cieWtLoxNTpIV|X z$Q$_g2R~!$TM;T<5kgSW2n~gB`b)RU_Nzx~|M#1Rm4{Map%XovFKXWM*|&z}$YXLke!xB!^HaL^~Ub7-@z`M%D{ zA82v!%moI>y(TKv1(OO~RuzGMKNq5jJl!ZDtu6dn2*o=ReExzgS! zr@qVhp3849jJL^yEa6@=@9#BpW-sbkwiB;xCFBmUn@%A8&k|QW`hlz$j}8gCl*^bZ z;*GXM9DgVXk7tkWSX2=R&E#U1`CW@GH`G?(Rkfx6W3yXO z@0})kZ(wsJZF}V$Mm+Wwev3i55Cm@>U^68NxNBmQq)etQ{B~YJR5?@l$d$mSB8ZzFL)!w>7`=ZRbng!#v}}Dg$s(T=!FiVJ-T+b!s-_{zvGMqis16 z?re}z;BOWWvg0@FsT*JU3F0T!29el@E`=ux!xEkG#fHr;q)DsYK}V=#G;XE;WVioc zHvHY`yZSF65)k*@k|V#QbbSfYrtxZAt+K$2-x;>gYDVX_z=v>50m;ZBp7T7cSd|L1 z3mmC9rlI;CVSRkN5ynw+7Wwwz@t(bCI#4|jMkCUxG14?cevf261z$zaui zxbW%{4{T8TLc3_2DW&gnY+tOw=dv-_Eqi*6c2J4&Pw0Rhe#dcV+y}oU{3{==6n{ zy|C_|{b{&D|M$MRx@xyh$$q_dGsBz!h#l~&^V#mNOsEtVz70+X^>;e0*=mfUF}%Nt zdVdt=c`wVsIXMP!puLm%A1CYBRePuWrLZ$WMp&8>a+ z*8>3f4ag#ugEa|m*4~<5#$Ob6BD#WQAmDH z_-!v%;jm%nsJ?;sbsMiC5JQ5|J9V|`pA^rAR@!zLHcxIYHQb>Y|1DABjj4anzxyVY z1a_lc{Sy6bwc%`?QzoPbYzRTGXO8o-RM?1bmuN@G#GP$kYu_-9g}h&9RDzNleXqP< z_-Ik=HL4jzub28JYKB+nEQUnc~d>sdIC3D!kD;N(f)0K=KVI( zex65U@mej~Se)koz0G%faZz=&v$AqrGc2`e^HMM|Uxi zq5_@c+VEVn7c84_^jqCy;ZSt%5h-ENzQ+H6{29X%^|dA6F^7IZl#s zZY?dM%Vz1n!)mQ?G?`rvG{`+KJ)s{k>sc|Eq)#g)D5?s+^Cs55wIJ!cg$D8X&rE^& zH`lkFB6Nh-U30Dn*1y(cJHaGuc$O%AIwI9?6LO4s*b6^C5Gjdj1NN`W>vc+Ztm-&J zJ^06Jm^(J%=$-#8-cII;d+wpda#YXy22EiW{A(oKniExOWW>At_HLpA`^>ToSHw$m z7fog<@^Y@F`?O5yUvE>{b8C>r$#7sYbc6(6-l<-`pJ&9K5kI-B&FdxZg-_mwbAH@& z{%Xl?;qx_2^l{=7iL)J(w>4CmRD3=#dDU)r^*}q;!d7n=gS}3vE8yBo-0F9u(gNI>_C}p zT1e&$(3h6R?Uj$_n(hjHp_g47^IC|V-F?O^*zK|7qbAW$qtkp5VDA2t6hU_M5B;Vp zGrRR%{x<#BW-;eLskPy;{>{E;W-80wl6pGb0>Xy5-e}6*Dz+jSehNkQ8P9^Y)h!vy z`!3)*Zhu7__6KXUab`J=3OPbGDyAcpx zr1@}oXSy^_fnbe+IHN(3edrfgJC*C(dc2x z8rVPdt6m#}+4pmgCR*R@{bBIGKypQD2|sM5)x}U-^L}DS<(m!#k$I;Y0+OW)3;5*7 z{Jd^~(%EZlZ$#Bl4l`Oi0r6%$bp$skCdl z#)7DMBXmr~&6I_)xglwO_JGd)IK7!h*KQ_qi;X>aj1ZE3OXSas->doZtZpDkyP|BW zmhg<7?iv}rNm9t*V4Uy>v>@1BH)-^1W4pL1Uf}wPcDEGAv%dC|6)NMqNeXwq2_wg> zWIe&lhE^om2pgTWBiWfIi0DE68x*wP!dWj%6h8b$l=HH{{}^P~u>>ZM>gYlp1uS z#!}b$E3wDG-pvQ&?0;Oz?*9On7$~qOPWUl(qTAg86*F<3FW%(SEHDL6nz(ue}nYOVRyqMZLrHu@yY5Q;%!2! z@yS|jA9`zTruDfp6`wXc>PW%Oz8~YI^Tz^T(Wjg(mKt5rF^2VtCPPM&>&4yPMY9jZ zz7_dFmas%6J?<+FT=H_qoim4kue$@G&5Je!?z+(U+i!AiASWikw?mPGQs*z~_Huc+ z%-;8HZygg$tbU#rLJHdJp6y`;9Qv(KcP5@;pZ@WNDGX`+V$2n7YO1RBi_mhGu}lb8 zxWeskbgmm3O!R;KBzM5$zS1)MPZ|~nMp1sP&rc$2l46hdUwS0A zCH8SYLM(m>JFliBbq7vb0CVmKoBb(yAq}EnNOKw~Y?N}}txNi>+V~_M8(saJ>jl!X zlsQ%bF^@3DD@AWWpQVAt^j9RszZsBHzZ6|b^^o?li>7#4F^o0Bo%wJjBW@QM*ZgX- zp8EbLWx8{lllBj2vbR>UEa^O4tBo-`Ge=Wjf;wMua5etJ zYJ6(8LhHtsv-r9UZ(05$P2>%WpgR%E_FS>p{QJLe$Qv$Yu9uSV_g*b4y0)~!EKafI ztYeW^4@$_CzA1q?pY^i}YVL45IHS_NL+==`!rX6$-?I74gLO4MFw9}j)4O3vU%bHu z9kaQ_@Y4Fw5?Q3*gT1U}4ItJ_3kwV({O}o!OAu`z;rKs;^-R945}Wb-gNI6K{+&tM z7r%fgC;~LXZx^ZeXGrU6YS6-!{9$Uw-Jt^ocJoq1#ku!FMk^?W$g*7@C(|Ssz@as! zwWPJDb*1%2l)KG3)`;1MeW{u%%D+E&Ls5MsP7iFpkH|94We!QOs;g-sCx&UEZO;fET-6Eb zn%_P?1^c9Pux|H%vc6I^hAN>MN6)C8MLvB+^FwFR!)5!{TX?0b(Lthnm{C2XB|lq% zt3td;CqT0yV~^M++a`IPjz(AYzd!ttx%x^d?7`QIz_#w}_$Xeoa(UGJJ$QJ>L7l#( zuc5_DowKDT(hUlmTWu_or^6~%^+yXe^3j`C8vWUhNQcq7gYm`9W>SVnAN)sGjryLN zsOD=QA<<`!#L>Q*4e3!GM)^sJjIDzt-Y&?(N+QNI9x31LNBK@Dc0rEr({&Zo zqMijISQ5TQM7u2x-)yAus@!n#7*G~?ixYA*8>OaOwVV-8&8?l4w>`G&T7j*u7Nq8O z$1Z2~i62ZbIgUvoEm?28eR94eCy8y7JbPr=PVSM|6->v+4L`B<2>+pSqn}m|?Qwwe zjnUOSY-MJ`NYqtE6I7gSsN>=S+8E*=oK;RQPVR)KJc-E zpYHn-agRAvpup_9kFCLUTQW~Y|K}E{q+qO$8%M;8+e0tP9@*RR>M$r(O)&pm{e99o z)h}YCmUB6dL?|8?)UH^)4iG!~AVHm+3O4B1Smr23?`4h`@XPKA67Ceir^NJOjflwk zO;q`_lXwLJ)Zm~ks%-Sk1yP!jVYw^(tDqjcG9RP&n?{lg?AT|KQ5pf&Jnf@})MwaK z6^$y}d7yse@RZ*JZxyzaf0k=wz~EPYd?An6;*Wic#&aOHH#PXQX4_`S>!Cnnekd5 zoF&Q_tO6IkwXt}eth=objD7)Dwi7D4pHRyy>`m=yD3 zjJ!3J$!*HVRclia?Y>jeCt*;6LsDctaBEng+YM7TIvvQ=y5Dqnsv+lpM4nYV-_G2|OxL=i15_` zwgO?A^ZMSLV@=M_oC=hq)|j>%$|2Qh^?&53vy$X&@HkYA;%m#T@b*tM zx>vjS`ovTS3c2R`ojuncEsnCmHA@zNQZBLei_!@@0G-kYe|tXO=efUVG6Y=G=d!7i z26@6PPW$cRFETK%=}!?#!gx5>o9Q44Ta#`ZHVyV7rz<%HXA7l6LJM27ZD=R72cA=( zafCCMfc?^I4}A@ux)WZnk)h44FH}76U5fZ=yVGh1b8h_Nx~U&AXPd|9jk3k0b_Uze z^v)++ZJ2RB68-b0Ng|&5L$-H}n+x$)#dkV^Ru%`Od=?V@pROPvwEPU$sIWvOI!y+d z`ar;iZsa5HTr1WMd8?eaKG+i)2Af6Moi*T+-K;=F_-b8?Ejm!zz1FD#lUTaF#9>a@ zY*tX`h;bNAL4}i4roMW0jIn~b&jf`&Ry84KcjbZL1~Uq8`;kfXTWYMK)+%R1XK9ei zrWR2>lGq0H`4;q%L2Cp4o_CUJ>g90mmxiuvG1qly#HEf4m91ZEgOroe@~<2y@?<(* z`p1V}f@t@!qLJL|c%`7{=xey>ZarUiN*RBr;jRYVZus*I&6y6FpIYvI`K9&W^?SAt zYZDVO!DB^UqE2JRN?({Vz;6G*Bij9Bbb1l<_X@n2NCWpAXf{E_RQUdI8dQh{%zo!a zajF&19|Mb?qf`>x8Z%({$?bm*>|;3@m#lWJxGTwHUt{uqWp=j7bknDb3HKcw0!HhR zcWP0S63(aCJvD8cmS2>woEMfeuahT*hZ{ge*U{%#wc8X@b71<&E_|VP{<*H#>E2hK zdRiMtg*;)~x%Rq!#Md4xZcRhzD>X_WUfB{H zpBYo}?;OhMZTjjl5*eN!nBJ<*8J$L{Z56K+<#v4-;abe?CQ zXgsk)FkP4Vs4VKSn>$&}7kOhqjzy75ImPqrXfdmgEmY_Of<&-(bLOjdPQnfmcjeS~ zgU}AIN41L7Ji6`;XUAzOD96=2W2b!S;f@JmyH~(3USuZtR)2sc;^DWCJu=w+Wcu}v zylwp+ovAgZphYNChca z&GDwLHhnw3eZ`z{u1pp zEckz?q{3YNj^@6;C61aSe2ZU=rY6sLFB;MS{S8Fbj_K)hOFe79;MYlYyN3~^2&4CX zzbk_Dnc;m?K#4HhK|z`mDkKInxToR)qB7qz1wIFy810C($mFT^=7FDVf8oqmuXQe} zpA^#6)XcA5`Dr3J(IC?1J7LqtM5Ap#0DLyHz|JH{#xhX?RfJ2Jm6KOzwa~4Vp#$no^^>Jl$FL5=m&2?H`8E3pHC>N)dYoZLU%4wAvVM|mx*yHQz#~1PkW#n8 z?}hgl8+GN^tc&cv@IXnBXX>1;Ots(is26K|AAYe0GN$rGQcTo_lGLBB z*R*Iv*~2|gTTkZLnW1(BDerPrbsUf8@z8Sh$4PF??n|u4T8jnnGpv=8Xx>s3WuZIPZlHyxM5(Zu zRCK~&RHyI?n_P%5xVOY*GbM1X+Cb*827S8zrHe;$<7CHa+=#})OyfLG^dNk^L?!z& z*+YOsgv|EWU{Dt5>&yh(mjGlS8jE#3_C~Zal@^6EL5xarTd|Vo<$-U%btf|7y3ET#h0y3y?`EeAatg`a&Aai0wl z3>99WE%#qdgMtAjcblg-TVk|*Uy3fLM~o4!m5u6>iSLrDElf4$Z7io9E+#om0?c0K zeq<|C`M&WorM`;F5M_Cfc>VdA8-JhJ(&%~*GRazpgZ%@v7y2x07U17W*VP9C8=#RcwEU$&ayqaMkx?X!yKeV0TWCM& zwc-63J99GdyN5Cri%sGjLet``yTSK9yPL_~JAjVlp3decZ8UJk4;+{zsPG_=B1uUjG; ze*;NEpRTCWd?0WKRgSc98q*KlTOP`kgEDJbl#PrP85C5naC{eM?z61f@NB>#SSvi; zN8})O--AF~cqLc;#Iz@_%5I^MlUC4jLIp%=hm~Q;1dsh6UMFl8iVtm49ZPaS%L&@d zR^C)-al&q+c9zs1+ogAAd%8DjriDj=MKU?v>?Pu^-8?(qklbv*1(lV&vo*eBh5`&^ z6y*pw+?9B=%dy4pv&=WyqiAQpr~qq}!J+lM?)R3jG->JPlNJ@I{JQn((|sh_o`Cb_ zl;r8M(#NmRQDCO=WoEcO4FFw)ykw{xQ8g#~ne%fWG`!_^CygsHh0;GZoo28)Sss+1 z9?ntYk*QqGuK}5DT=wV~WQ8r+)+rGEk_4@o;~R~jbXD&h3sULzH&U7gt--PUCg1k;H9**A<~&e7Z)ZiI>e-21@fH~q{q;CW1D`@A z_Qxb4J=m75Hme#;`De<+6@SWCLJO6%LRUF{*R#rIUJ9~bN(oH8>V5?9l;`1ept-DB z{b{eLIes9Eqq@=c#ElKl^W*UnNZMep_=cE$AxI+HqFGnbJ+j+PL_D=|9;YYll1Ckn zN45Pjq#hkim=#E#Z9zd^Hprx*;W9QiSGY}q_Bt0wUfR9Ge@_f|^8I4)kkY8-|Ljh- z)uL*`a^g#P@!fMXe!t^`B7=poA68DCr2O*T^GC*#9I(k`oe&<{J%W^9S zB*rL;pvIbWj$|h9o!MeNs(Pnkv`q0r$N#*_BEAq^@^NaUd3| z93vT=bx2~C(3f;{61Ru(Kvt`$tGILey-IO{Ngxb03IHy^qtV{n?&5J$h^R3g;n?|R z;^J~-l9~7sU7lJk2L)NY!~NwJXlgqhE}t&e4`gx_a&5ZeiK6weJuKAUSIdNAmz562 zAuIw18)z^1=>cS?wQ$56=a|L%QoNa|DK)odWQQS$9>*0WM&oW9b;o;{t_AlgI;rmC zJiw}^ZX)^oK+i1cq>Y?Fnr;0N)=Z${#l0IM>y@*vdn=y~8R+Tj4`+xvb6x2bU02sf`llCF zRZVejGBocat)TXMpH}+XFA)Q}o@mm8azy5hIZkG!s#qN7xz*{uoy8fi$LVjju?&(~ zJN|HyC_F4s1`R-Lb{dmB-W>vLngXpK&fm_R-vULf5tJI4K~_(D$Q6S9m|V3exA! zya>}iSi&uEZ-|=_P0|qq8u^77Rae$AQK5DVzog5~;C5&h`E)+bWT0J)Wql*)@=F%D zU$0jY8fC2Wt38~pM?P(890C*HA2T?h6<8~lJT(C*fex8bVew(I6;C>l6g62ckIVNy zC+~d$kF8$405R>uWVHfakcwOn)=`qj68CAvoD0Y2a40P@*FrH{&ZH7cH`D06Od5R= z2m~~mM=<^_i**Cee&QWQnX~~b3q05bKKH>rHXQdn#;~~TDr2G#T-{85=V>_|TU21+ z=Vzk80xrUfW7XU@dCQtT)smg&Oc)S^Cy4#n$ZFP#zq*9PWoKn&<^LR{sY(utd7$OE zKMo;78O1b;V(gdn15a1f;Bg`jE0X6&N*HO(M)>pFyU#?QXAWJ+0UenM#h^zSaKkx} zQiO~KL2pCVLhPE`OmcpC`H-mNO>$a3$6S+6kdwSlU9lurT(X-nQvqg=4xo{bfT@5lva@0mgN7*D72nmc$O{-@^ zMfFkX9m zIXs$WJ5=-Y^F~J4TSVqy^S_X99{1$b97WEzE>G9x3edl8njBZU&V4-P2i`NDgqrlV z=&Qo2n*+wX@>-4Z5#|S?a8A|S!;u0{hn}+01+TXcVa{CpWf;^-y$AM+pOlC4H-)=@ zvHCTL+`wxN=;bZF1j2N4l|q!N|t z1kI^EMR%L;SaCG{n%%M&sk1~?TYYtZI)pjWO7Y`Z4fO~d$MWMNZ7`?x%H8?Q#xr2-$ z)!UQnXoQm>iTw-UWBZ{Hd5;6){CKs!?%mbVzOzrI0Hhxj$+1{KscB0>+g1T;Yn``c z@*LAwl04moWh$0%)=L3Ir6`C#0TF3=wS3GKX;IcuOn+UyF2O7_Ki@LZm94C1pu!4$ zdVqxXJbwJx3Sjs^7Wamj<6znW_Jcfif>07_d@=x=mj+r$125OnK6voJ3#7_{U3Zu6 zM{VGo802VsUeA3CI58)w=Jvv%w7??D`hO8FACwucKx+_K-E=Y7NHV&2u9Qm8m z03I-ZB;v;JcpCc!X)^KQxpUC>=?XB&`zLqFK$JTcEj&6rJxeMVmnIdO@;+v9o(h*^ zdEgw(Ao%K-vjYP2*uRiR=n|*c{+e>u|XxUZTD52 z88j5A9N#T701q%x#RHswL3}5;s`MgSe6QenWCGBsBd8oqTX`L8mE%0`?N;9xzXB-H zLchs60MiTdPVGDB^E0QpXdWCe|NdtOfDkn*j@Ij3jsT3;mB0kGt8DLVKtdGOYj=9h zHHZPVyHM$Y942;tGSl)yw^T54oP+Z5Q51$ry|e(#f4yhuNZ;5rGoq&V7)i(QG$qV~ zGU@^}#2)SR2(Dj5(0JjJ_p8W?X;LG}dIEX|hVvcR`O;Uh%Z)eR4AX$d#Xy!g!QI)d zN)JwGm<4+n2*n?z9JMgqEob+etbXU89>C99^2Q7F;K0~!H2WuRyDa~RVU86?0vR0q zH8byCz?Ae`$URQZ603NL^Kppek&G3cUKqAoBv|E4`}UJSI-vOMjp})Q#<&H2)c#j; zHz#+8N=pVg)|~1Vpp`bHj!!2M3o;fSyUl%YH?t!(B1x?6fSlgUNo%0q81xg1`&nx- zkFL)Cy!?j8(aal0y`aCU^*@B8f8kWkt0K?)HC_~zA|SXBM#Gfe+~CPDvSA0BFBZ5u zu5L?;3V_pjogbsEO7?zZHMg3+@+8?0z5rRfs~l-5YTTd90z@((Y&C$ILD#)@-^S!q zKNFBCgfOK;`88))K!PPfEXqV#`#HAHmjL<$lu|1|ZzMog0GPsI^>^KNC#Mxe500lr zLi=9e#`;huRENOh$S8*OZ1wyEa#lG30EX9D39N|Hc_9|k%4qXzI9ui1^%$c9s24~z z!sey%Mj_Q zpHw!4-e}{Y*Wu~LfL8ymapD%`T`^E1L$J~mSoB4o*DWRiJp6vXGm3{>-Fn?+vBz`n zIwnlc$i>CMp{%|zH}(MKH1LduH4kZ7_n7+YIL!CvB2W~ZMt;7#*Afd!GI9L9{$>q! z>twT;jK%X%zg%)|%Fn6zekhUc-s8~(+s^yiR&q=Z!=U7p$;4#_B-&gjS)8^|Is;SS z$mnO}_O|@Hz;5c+P7f|p0;q``pX~H4vP&;3ENscKBtFz{0fOBt_IzJQrn9T-0Bt2%BT`12MIG6m6|4-)ik^jvk~%m5 z7Jni10;1@CetOWu+0h-AB8>?t1Iz-5dEP58Z&QT2UH~j8)kUduUv%w#Uph|6F`&7c zdiVO5j9p86FObTpuZKxr-9GQQtN|-*$r&zmc1V-tHJOA%Fi%D0X6O`sNl{VpLbn-w zO6>FGJ$S`_(v`o4?J{n9o}0Bc-3IKnjgxQ)Uj2l6aS$wA|p(9~8nAGeyU z#~`OgrGb7we4)ZVO>>>KRd^Ik}fQx_a6J zc(M(48H;EIVZyDMMDV7thlK&ydCBwIo!w6)>aRNAL6D8E_^83!UCh){8dL&U)7~@c z8#_udI*!BPax1Ic89%GBSCMxG-UECqvXtz%`|SGbx}}68aF35Q%3*XIMx9*VC$^dk zvS86Kbz}AmW8loMEly2+`oGpWbH?;C*7;D*sPZ_2?11N~`FD~Oj6e@0{O7jkh*2F! z^N3N(_UmL`#w88rH7&XoEhf_Ax^U+=e}mhT&=S!c9|4}sVyTY&iRR+&-=$jApxS)| z1cVDZdFbten0SWBi!bUCK%VSbdAWf|bOFbU4BF^J41r*9VRfL2;I>|W3`&w~;Um8K zp~4_g11L41ceWjqOONzAKfT_pX9cgDGAo$^eW<{l1(`g$AIZ8@A2YitBnWM>fP8VG zJ%X?^SZg#30kkbbNz7&@h~4&MSSRq%iq(bpb|Aik%BP*0u1!lL0ZaFam^!n~uj>~| zB*1Wz1+)Djv$YOj1IO_1HMG zB=_wXl&o(hZO0#u)Goq%#SSho+bwX-Swx2OVu`2|6t2cSz?^vCl{RK5fDAiBl*XuNypunarn zpZ_L*f&TXcTw4#tfj|hln&^o8EX!y@{&8PP;^6(e;V%W{C_}=XRrs$5u}5SFBj>kW zP*SkbOaT-)Osp6#BYk_apTfzg{8x`E=)L>^YUI|s#fF4>j=|H-fwS)WXcMGf%-3|1 zUm^0RxBSc_aUg-)SL;u7`y$yY){zD9a)hz0?%sES(+>|+df-G>{3|v!*5svl za(5GZ!SOelFOI?fWwjUmiFYr*`D}DOC@B}mc#*5nr+bhF$Zn_rKrjLRE#9YF0{{>R z(0uX#mJ`n6U;NcQeiMB2RD_ZQAXcFu z4}$_~m%zn0|Ml-gtr1I~fK01=H2Jq^*EI7{nhdmHtT%w4vYY-ybGEmSEC7(^q5}rx zS)*fIGO_KonoCHcGmv;d{2d{@Xp*xx$1SEFIsk z4HZAG1MlPamcs!A89k4l$Dm`Q-gf?F`NYT7Jf7;MR2*FV%^&~lZj%?GteZ6Y=oZRW zl_2rm;icktD>tzb9br+$*ROzjeO4d2>0oC!!~iNO|)Skv6+Bd!)SHkOwWBbA$c;-aFs!$R8FP5VJ@yE)uP~ z5kRiqYNu~!sOaLTYYym>e!7xcqy;*gU={*N;cDU;ijuFXbzBkz2>;mvuo3n+=-;_@ zXW0daw`+53R)5V90R0Gzx8r!D1DS)Wtjk%#w1v8jpv6x%zT6D_&y=2O!X(Fgj@x7x zyIy$T8k{~S`ru_rO$tuJsqu#oj{fm8SDy;rjY!d}i}`B`g!#249F6NFrJ`m`yDmSQ zeBL5VgV;;ou{R=JPhKDC$GrMS;JdJf;I3yIOwmt`q z7%F35YZ0$&Wh+X`c#i$}$0`7_N)eUTOk9IpweeNf7|O^+Gmy?xmMautYgoyneDk+$N z5i4Gaz>Sebb8zm~@@PV@B?DfO;w*HcZ5#x&W#`vMI9EQnZHAmBrO2l~Vy}D^U zAPAX0C%*63hPj5VeK+mhHv6SDLUF*~pbN5Rp1bq7AJ;9iPwceK>#$>8n&7R~vK~hI zO!-&slZCc&kJXwGR3qiXzT~ zYuiN;O0=S;b7Aj$z5(*iqb};&nn;N?Vfm5eW!E~)kQ%`(>GZvPiGDJHTEux}I3_J- zE$0F8;}UjKk(RhB)8B-+Wgy}@{-3yck{NiP0^uxS@Pn7`>fHh))2pFA2>GdbV{m4wvWMHGF~_A#=esXtq~6}S+C)YN zMN*L-a!8+fLj{v7zAcx!ZV8j{nKKfk`Ti@@Ttqw6r`Bla*5aRCmvzZJ`f{ck8yckA zzbP`bx5;l5#f#={H@#Xo+|$vTyVV z#Uzmo@AXluV50|EmH zkJDS~If}8F;n`7f$0qM z$a9-<^RCKrK(sS|(x7UKkNS-C|2+C4)>U$Gg2Z9a@zc6tachp#I`~5%S%$&2B`L~t zeuue}j=nl#mXQn(8GAZ2i0=3UgmoklC(DVbKw{SsO2Tx!Q<)Hk8T`ig20$x6CRe z$ZEuiIwU;asBI2Vj7y|EkUM}Kw>bqQ*mRvLP40A^5%n_d)o6VEs3d_6UwKh6ao#;a zXJBTPX#eX52b-S|lu^Pa>G75;$3iKg1D_JD#QuH=LRpQxTqe$-1B|y(tdWdZQij$i z(6YPoexy%jVzv~#`=yBgRyI4iQ_J^VIluj= z2&ES`l1!wdTAU~%N`jdRaA=%*Q3fU<&O6E~LLWqGfPJ=~k-J)uu*a-tN$w?-3v>7X z<516)l=w}H9t6Dpb@Rf{Ldm&wUU?bV5nHri)JMo~~kr^qt<0o%m3P}|;YO7n2d#%A^;!S1-N-ZpGCd4#3f zhQ=cE3e_6dz$Yc$I>RyHMMP=eO`+{bzBVmBn09~M{0qqu1f<^zR`mJ9ZhBB2gFYG3 zD%Ys4=w%&$-VCEh>+NHKaX-@U4=3Y_>_652XL1sN0yD;6?$(0A9<`TszJ+ooQ;kTs zD#e!BW18Gr^Xfg<3QHD?NUJ|$E_YQsXqO>ogx!McCySZ&>THlk3S)MK1DE)4v1H}B zP)W{|QD%Pk-6iRSwyZ>O2Va5KSM3mH0U14dd`ozy(F66pkowZQ#ARk#evKSmKRJk& z>!^D+E?i1fG2U@6a$7}OfUiujb~ z?HW5GT1pr=zdc2Uw7fC7qR-yvyf8`l_&_1C>Mc^^SSdF4U~-FZhkee(MUcGUEMB_6 zG*GiT{^2sU+}85ptj{}y2Kkw2p~Z1pS+}_r_jF_t@58BX{MrA^L0h^@dxtC(oY$J# z!A_*>3dAuw+EzVBO2=3p2-I)D(D?29qigSneh=j9gViM+jGJd^ zeDMm-y(iYMbxDGm1q$6vyYTc?8n)ZBzK1y5j{R}W1;pc6^;@7i=<8X~H`{0=l}DQE zo2<)Vlm0^Q>zZTMG+n&eej5DZ9c$7#-~1ZAeA$%`kMOViCou zeUdIQb|qipY8IAsB)*TFJFQy$Zr5Xe$|ABV)TnI0vtF@Vt_AVRe{f^q*#Y`-1$ypL zn#gWT!-{g6QaSa6xSeuDIr){wcUOd5md0tt+0xKP{VKf5(tkdyz1fGLNc8Mhv8G%K zlX*iih?43(ErGi%j_t8AlOlHOzAhyY1#z>rC6&eC+8HRN)7$mtyJ*Lj_Nq)~-t}BM zJ3Y<9OKlUeDt$@T$oS3?FaSZa7WUW-%Tz*L%Wu^>EB~HS?ETEJhnc0QLsKNdLgV>_ zC_A(c6(^Kb#~=&UJzi91pZqzKAhbv5drvVVOu+gA~ogsnoG@PU& z2jf>a%_GjG@~>Ea5hnZd3OVhpWhe<@eVBu4#{GT2r`d%Jo$Vbg-Bo5M{UbY>FqfvW zG@hiQd&2H3Cw(;9BD^&}dKV?D~tf*hA?=ExYe8qJ#@P+vm1wj3H)dyTwV~ zqO-FXv#klDa9zA>r0}_pn2eC1ZgG#M%103p&&RH<&5M}jiLENvi6}crh=%wM*DvAe zfKPtC33Z0LW~mHPNA2=Sx@U93eyZoQ2{!GU%R|)rA_pDNMZMw;*;H~>^bl^=HJErM9SwbWC69qq;(zI`h9nQ4y!-K zt`hxIpu6}5{gQ1EUFmo{GI@$Y|GE^c;yyj}m)(I`FU7W-D2B}+tTaZ(;##ENA4fhe zFtWI8ET!UYam%iaof|{z;C$l$@8kP29_m=w7(X(!NKk_OeQ|;2|6U};ExAlUpiYrD z>O}^h1QfZ*%2bL}Tw!`&FMn0?Z^_ChYg$@w1&;78+4At#^9I?+-ZY+wbunMG6;v z&Ldj0obddIKHa8^Sf%P0rnHp-0)olQV@@w`bLic?;s1aP-q#$UiX%hulHpPfZ>r1x zrWycLl0@qf%DXB0NtGiJjK0%_){HIU&y!piwk@){IC{jF9m_7{X z1*&(a_6}<`OK{mqFaQ1CgA_{JxR3r^X8o>DQJw3#ujpdTBa~sc>}zzS;4v7Vt-F7< z%|{!H(J5t!agX9Ni4^{4dBmp#iHBCKbL3-;T!yv!J8pnr;r|d>*ZA&lqHKAU{>N+T zAOD#|8U5A8CM>ddoNo=qt;QP_dgqO9g{Xp-RD~;>8kOqB*t{**2_q@VbMpUSyF(w) z@2j$0ie7yy^!1|dp(u6Nzy;i5RM@wAy@>4}Q35>PrZihDI4S%Tryyp3+W6VuGZ@K* zxg0*cz}_}>(C`+Vzvq$8r^;~$&HZQI9XkQXlI#P@5#HnZ4Mi+#4-7Os3k2IY^8rCVn<^AF8EynOZA(B;vz6#q|7o-T$z%tMSlk zEcV^w>px~$OY;&1RHWw6C|^2<>c-L@e7Rp?yI@cjU8Zfi8i z;(-kT$GHA@Dggpcc$SUP;D*7|0tH(0F$L>0eZP*<0qN)D0uvhht=p z#kk6QFr(GM>_hNZ+B-V9Pfh;onX5pjF{?K_TpB7E3>x!_&QqcPSS*NTv>OAi3eKD@ zZY`M|sv+&c&dl}yA58&yhQ3yB=yD}|w%npI{8a=meROLS$(H)<#FrW0f>0IN^b_!0 zNEMp-o-=wjY7#Yg3T zb&TUJ$>8nk!Or}7=qkFd9o80IsUr-Lk5S4JxgtSzGl+O?r!Ntf8S5I(WOo()O&K%GKO>Q09&_id@;qvREq(R zZ^-kLsBbg84I|4DE(8F4bJ9_AAHy!|o;`nfglofU-c?t#c!L$nGGS-!d~(~ha{60@ z79$cueA-tU@ZCNleo6S)Bp0FssjWiUXFG1_An(aFYUh6(HrKuW9-4%I(E9vVtR=XW z;L25B-eWOKp0i4>w`?Nnz;kjA_7w>cdelFOEc%UnSm8#T(w@6(&u@s=hq&)^k)={- zei-gVaL~wvEN~OK!mWZ$bGSmglM6TLNSzrpUw5W1-+KBsplwz*=84zTo7*aMZ>zA% z@60T$PcYexw|kD{SfnjVj1)qQ1QcHy~|CUZ+LBxW5{RG@Ojr9&Kx8|(YgH+Y5Q2$Wia$>YA+J6 zw)DCeS5$Z8iV35k-}d__971U^sRC=UTThAcU+d*#sj&TCpfn{<&rUm(4r4?gWCHl0i}L_SBTMgRKeb{JVPiZxkC=(4K+Nm%zw()l@m z@i|C$rffo=^vuOCzrbZ>RB>4%#@YojT09au2f!RKo-hCs6ZN4 zz4i8|`m6rj)1OV~Q8|h`ZuCzu?_%)c41#=@d7n3@;dA}^)RtIh*P%XOs5w&At96`% zN8-bHOV@w5O4(Xw<0`cR#9}D7ELY;iX&NjjU7ypHsYPdI+TapCy&Q`)*oF zP_gJgYo*lsx+p6_hPR~zPUu_vUejl_IwRpeMen=mCnAZK!FMZ(_q*TU27^+)w9wz+ zgZFUM*a9DFl*iz`!X)qpL!8?#$&44;4q0mS6Wp9up(9Frp=BiXUafVchS#5zP_4xC zo=yZg;w8BhV9@d z9-64{ag#ls|3cH5BhXJb{r-v9@^#Ypm-N$VwDfSw$Zt&t-)*=<@yuM^iS7N?6ASOp zSZ$wn~&1#_67=uW|9v%9tn{#j%KXF(;X_|xJpK~;O(rUh(r1*C8&4_P_CObuC ziqB2o9zL$b>c88`7xCIgH+{g5|ddNoumyzRH&WsE+! zNXIV+maiGJjXp#fMCr-NPrad3;)v>g5b>G^5_xQ4E0B)saXBp&mnY!lto?o19_)=3) z6?^~j!Zmykk2&%kA~n|?1N*Y6bS zJ|)*~#%J%I$+A$G>r_+T-bt`Bro2IJF?{1Il8O;;wsEZSN7{W7>!)h0dHJI%qoI!} z*A0{>2lu<$Kk788IK?m$8A!iHf<_{ZOInc5bshJ>PlzXUN`MeH_#!!^gUXt3J_=RhXgIaA-Uz|})O7V$@UIVXLc4|&tZg&d z)Prk_GT{1_qK}`%q#jIv4FzwS|$|J{2oSsRFz3G#E@6-EpAzT5WlOoD1_t)`Hbtz=PV-25Pvx6#i}quOT(P1@VeYR%Y#j^1Z0s30?-y{GE76wwl$9d;;S;`|E{SeH(xqw zJNcP$cqonWUo0T(gFatH*!OLHi?Ly!=msb?=DmX2Mk+$szrNAAL^L2Cpz=)WNS{{; zB2+jv2R&27(f*0pPMd|YgVag8;dR$GM^$Eu2GLEHyE0_|rYxwF9%X@s?7|xqix5j_ z#8wOOrzUZDQyq;o{z?{+L;@T+38IV(T?&mb3w)25USB4H&*}j5vam z!ZhH;8q6%q+l(1=Q#jy7{+e@DV~&_1t`Re)<`N>^5u_lfBFxvAAYev5qsb&z3l7iu zS#`vm@1Zot|AqzxTyQjQG?wHQhy%w{1r!iNgV)CB0&GC*WA=CCfeqYb2!)#z$2~7R z^sWe_t$s$-!?|jKT1(#yU_-VjsNGeXT|5D#4>x zM5+=M0CSiq*{u>Im`#q_O1w_bE3c1(;H8%?TajUNbthN-4xHD zZ9EzG;m2ZLE1;=biV00Y5dNF#LoyoWgyUdln1bMR+m~d=c9Da znM^vB#_g0c=gUb5bj)!$Adw-~F_~d4sND#TyEPQ87+Y=0sD!UG+Tv*cxGBatO$Hav z+EG!gI2^XREP!bPxui9X%`2{u|6JvTt0Lq>xnM@sQ5qSVV_I;xmSS(58QdGdFTF_` z3V3iEtCm4IS@PrN*^fK1v}4*sDSA}~8zONmi&9n^@gSfH8T5H@PmXyf_q_m4({AV( z&fuzPFwv@#+6;7}1(|yHFq3JFI918zO52F_4|)kAMZh$v(a3;~fqQ#;y|>V^(txD{ zR#CXuqNOC=Ws$~5DW09nyCOL2#FPbQyMeqE0d3KiX6SdjEmSq}jGjDnfoli^@c{RYe+fX5b1#kBT6@ zlp4bdxVwi8AHSPB3YENAZhtgfdhydR6g2^$D=(n=v27nKpoXL#j+iM z6UpEP1{JFA`pI53hi)la)(3p>)zDI^GiUu59y0VNTbx4Jy|AG$s9^|F+-7{&zOh=9)szJ8Kh;S=?}N=Da;d)&5j`hC;=xbHSJQW z6s4&O2hhIM7Swz|H$-kXqmBHk$^uQ1c)@OED!oHnzB{o%nSp04xgCUipX;V$IXdwij?`REg2!TF@>Rv z`n(>@l@M;oZE>-=kwXShqFKM~#2$AB@&MC1EGV0~i6Q?j(HzMP&wlh3)gw@L?i%O} z>ItImmSnFaGXyd!A7w|A(eDPbIR@~6Du?U3hKMRJyHG;vtwRsi3-*;5gHYwDS&yr( zDQ3@=0x{&+n9L|!oj2vuE=+J7*&&1Qtuf%*=l1li3azPab|lSu5Y#7TOl{SQt7mx- zgQ%$`&JA4%s#ZuK_2V9@>`XeNUU|mQqfU}7YI}>5tpw_32#!)Xx*3(N5SVw+Oc48} z#|25(kba1+tyq3dDb5XDitqPM&K<{1Gz>i&Z7GOYGLViySH#>3tuvVSPgR1HR zF{6sG%u`Z5X98mXIOD*O!D>I?b2lpym-Ud3$-FTX#^Z9M`2v0jA}barrFQOO%o*Hd zM%lI+=+q9_KO%)8yKbilm4PT!76U1iG4?$kR~+*mr);BzX2VuX_(Qp{%;rm^h%?;l zsRxobl^I>JZa2*x7*O=Smbl(8q9>r)-MSqb!IrQl+XUMt4I!z=XKe42THnl_?U- zbZeGLqMKmRNcL?M$DqnW5yFrSsB1$ffYl(DVjd67wN24nIqXKGV7-*VeHH`}y197) zyEwc!foRLV0ga*7-i8ze;lGGZR6yVP%qqSlX`@u7}3 z3nuk5%la%fsRdtHt6{2xT!V~!OaO0a9zQoy{2jkgX=G$|$ zwp7U4B*gy7VDpAM!-|j}E69l2z93svg7lt=oGBb@9BrZ<4=^8bJX8@T8w{2sShHe5 z(j0ZF1K3>D_yJ}Fz*yBOR>z90A+wz zPKCm$Kc*EJt*ywdaLjJV57sWYi%`B=DGTw-1+Asy)d_4vxS#a8Ya64;Fe_Ef~nvCY|(4tbwsq?as z4~v4vW-P|YhEd*hW-uHo!~PgnIk958Ap*S%BN6E~wy2jcTv#Y$VES?UjjEuRW78S+ zoTM-&MSF3Kfjt;Aqk?C5}v~INC*dm|h-f#_Bd@ zdUZ}WLkbuXv~F%BMJe0h>?i`T+rtD~Gku6Jrw#xQfxcm%>*{FbaKM9VF&cla7Tg~9s?k_t(%Aqh?tHJ^F zG>7*DNkP6K_5e*>YHNOJnF{fB2abWugk)nc16&?j@~dL2MkO|Lb}yNUxW}!(=Ap> z9h4L9dnzHF3Jk@D3|Alw=;mm_vY-#sXhXB}Q9)(c2Y=|gM(U~xF{u=&bIdGQmXH=W zPSbjRW;ll4l_+dl={QudT$Gf_~Bz!vJ}y6vY~dCK~ct3j`&VPc|Xy8LfVuAb#9Aql;p-z`Y2T zrZZzmh;8JEVO ztXu6JX27Wcl~1yT0sRLF_3sh-dcA%F+KgX_5OiY{04pMUW&jpZeXGLj8cFM;t+F6d ziXDerzpsD<&@E%O&f~!3(~)6N%n6sc<|2^2u6%N4KkT3W(<2{JUpCRxao9S?uK|Us z3t_N~pWQczY-Z>@eDa1i7w;P>_{_hV_YzCk~|Gr`fg7~SZirV%+O#p(y3Tv$9K z|L79hDi5TBA%i?Ms|^idW6x*tvf!-St{@n*^iap^LT>rijPgn@6coYm$I%L0UK1xL zjZ0%t)?8|xy2FXbJ1$gGoJvlevwsdwCS}A7UJ;__PgDW9ecmAu5DoeUn-LXH)m$x( zRA9#KNkBch#L9>5?ewUiw=1wxRXJ*fRGD561Qc93TCO%WKoFrHa;~^60Drio##@Ol z;XuXWOa*6C7QiW)#^Os{z)#=mF#TvCSE^uOq8mfl;{JkdR1k_5|9V%1aHD}Dz{;Kq zU~kAgx(z=DLpf1Jy=`-h%78##RM^0j^j$!gYQ2S*59;O$Jg#Hn=j;Z9)OM0X3@`L#5I- zHsw_&xXL%r+pl-?_R*GN13Xv>12T)pTQ8%gvZZIuf=RD@|;KzaeHcvL7mq#0V)tOrnC zs7!?F+_3{8gtY-;q^Hxii0ze0jT`pGlWFufb~W^>2%vwWuzf^=@Oeq!igM>`8&UA;F#q&W+c-d+YE?kWN~|AB3nJAe3QsPoEcm;@lG&6U z7Fd5mB-S=XOOH4NL#x4%zi+z%GaZJ%dLUgiE{dnj?{-^tOYP0)a$ zvgB6_gG$DM6i5zb)!B>*0!h`oGK71CS}e}Bqo%ry3fmZ=0nIjTf@(xr(3&NdcBt@P zDWUKU)L{@DRYSH&7|^%ijZMJOOCMA>Uy%xDMQb9l%&)@)83VG9m1Tm|Sy zFeEThy<78?=sJ*5L{gM)ws4j@~(&N_- zXmUF(xtVAuPJ0LwI66k6fVV$)qh z)v_Q6LN1-CfWB)UDWF->sGDss<|;^G2tmj_oJsCy_pOI7Bh4TkNp#v|6#LJYmI#7C z5T}^AgMM#IEP;E)(s!^N1aWh6_LyN{dp!@?VEgr^4E!1$GKosS z)S8<@YY@!dxWwGRkvq~M@1kSOJRa{W&qQ_ z`l15rpA|-&A7D~()C`-DID4~W529N++Zl>sGLHz^Bl3g$B7|ji1`L48K&pWt?lYD0 zopNL%4*5O7fHy)4R+?UE(N8b+>iht7MN|$9q&CX^$*tX+9m<$CaL;V5nPTW5**x2W za1uBJ+(e=GVp)mI+WU_Kc=q``J<9^Dvp`hv!g0bs45VkFas_B-6}2KP*ni-;Jj6G6 z^nig%Vp)BwLiLeElwMFR3_@|{InQh`W#MetK?C`{^owAaQLIQTg+((v0TTq?(AnTq ztu?U_)TyA!!a>qa)HT!V#0bLvxyTMUCpe=9dRK&S|BR+Lqr(BbA!aPU8k*yrXB~4P z`J*os_9ELs5Cq~#sMKnJCsQfh6(bM0_dqDvJ7Q!1`LV&XiOLiqH>eX1f=p00lno4s z8D~xc9`d367A)Et(*EAR36d{=$dL6zXZTZ=$RyfF8I*$!4uZjpdtit=^l(R!aYv0N zvnw1`tW*pb5U?Wb=U-_6P6B5~WwHt33TD6rvCAJA;2#F0Ua*z9$aXkCm}1t^Jw}c& zsSJ~P9W{v#to&*)7~voFLt3reAe;o^q$d_WH;s<$+r)Nnb|{E8IOPxf$GKq5fZQSy z_8?p{=(lA79|Q|VqV|-F2?Baog)p2UcsVRrCJX=(;%`Nm20klu;p9l4gP_(RgdwcV zdcgDt6>Z0d3?et)NRlXZFf5P||8|*!0EA+X0Wa?woTfr$`#*pO%rcFUINFFk=VprF z%;Ss<@Q|r2qt_>~4&giVg2Ke-C0Pw2OjA@AB*Cf;0LRef3Cf|4KTL z+zb?p8DZJUR3;!Bt}8^IH)Pm9&W^}92$IO3e+}?Z9-txMkpJ)r*uXK9K?TXLOk82C zk;Ko9+z|wW6y1_~cqGX=!CCcI(hhjNZcKkXNx{mVPvpOUb$D%E3pXy+OuH8|hzC9q zA$*!3#%2TLkE$6CLiyO5hJD~PR#@{tqkFY zNJ&3MMRb&`^ST+P$dhiPn8ustV8w3by z12P1QX|7C%x2mwu<8e3AR&1>yso>cnutmS>5Gn`4f$huG`x&%=wDl_X@5lf^a_nR( z6Oq6h&g`Fs3CwD#<3UX5>Zod3oaV*`f9T1W_2AIhLhc+nd?6BuM@*C`tRobcS0$ni z90Z2!{NjAl%-(PYlw!zFe$=Xx+0a=A;nWQsn_p$w3~Xb!sBA>Ae7bPgGxkABgMamTlH916e)|G2i?uAu;ZB{F}>XNO9m8V*HUPd_oC46rwo3NnT@FvdQXc^LbMjT6m} zt#4HjV+11w9^enDL#m^Ci#_ZIh##Jj31oxgg8?z)%t>(nS1ts1b=)JvfR#5QM_ZVH zCI~K3BK$!k!kto4eo!;~l74n|SF~pALDbD46{IQKv7Mb&5Cj2qq5}Hv`JeJ-2e6Aa z7Ko*R7fcwq=WmS0^XKn;IQ|kbV`WSt86X?&$Urgd1;M!-t6LxaDv<9^Hwa|Hy>)Qr9}R|VO~RxSp| zUo2)6r*~zDR-DCkrUo>9pGw+0qFB#CUD*}T1wqxJK9L>s-Vn^aDuVUskZrw-XRrKx z?BfNuJy^|qT<)e=95=T?pO?N!1`m0_8~mAjV?&RM0P=%s!A%zuG#$GHSgVx>e6{q8 zxps{#Uac03*^H?YMK+irhC*QYRE7L0D_!5zdMnd9WKiL4#nu7rL&kx2Su5e`VTEJPBf^xI@Xzj~wGR-gR3BJMnS z)aX_Q9jzH;1%nZl+|UyDdFg|Ykd3%|=SuC{?lRkV+aLwDb!PB zT{FdoOeHIzTMDTR6)e^Yj$b3MaP+GV+v^kkgr+tn3_D9Qqfp2#a)MexcO`ACDxd=- zgfYttLT#l$aI`P%3pQfGAp>J&$d885_0xIsM@!~*w=z=Mxe6&!{2gol!3Hx6C zXpAeTi^>8x{$eph0zi7uNNk5P0ojJ|t{E)Fq;RV)3xxTJm{AG9gf&*&=D)fKHjXAlHI=x$(Q zz*NU)ZB-IT$q2pvoWLyZ^|l?P&6u%hK>BpFF9XG}7X)DG=~3HDY3Fu>$pw=~by)zj zJFc{?!bGkw)r^Rx4{H;+2w6X*Nx_s>aLNJjgM;R);^>-DXKmG~*keXTx!tr8HeYfn z*sKgn=Vt@q*n^qs93VEokABv~)4VAN&g`LcDIGJt{;VEE(>k>+xX-lhuOztPM-3oF zeVvWa)M^Rrm7kBUMpa&FG#9;^6i=kp{ZE>HRhJhOks(54z)^bm^m}OG1$%g)UfT}J z{N7j+tEKkVQ0bVtuzc25#o;sMCi_?ZFJ0cEr{JIDUPRi@hM`49S41 zLXW}?kM`F-5E{#3EMXyt+6+rRTo$SP^4bJI1j*Lh$ak8rqUfticL_8SA2Vnb^N_gd;!gue2p zt%`#r=;axha$62|a8jnW$85p+bkv}3<&CHrgvHIUy6dC&UR&ac3Y0VZeFb|hp|imu z|F)l|Q3w>Lbt%f!Cv#J|dVDTLYZY}ij#}b`(D&3+M)eN3+75||ywMrp$jw>Xb`;b( z?#U6A-=~F1n=x}i0QxOtY%itUDm^cvv-xRLg#u5@kQQQtFx1+pT#yT-QcMuFhtg$% z6MHD_%>XYDs)YrS4(@bvS+p0HQEKFa+Vgp6U8=8((`NtSQjoWHlqDaC{UEAaeM-U9 z3C`NNQ_03pJN1`%bOA4=i`-5EJ~tL!kVVbWWOFP5IfQo9ZZ?n<`f4=q`>qI%hAs9O z%+!^>R^Vm?FW?#8PuyHhyr4`XEIK>jD@7fhF07gEhDpU?%c4=8aiI`CY3{Qwn z3|MW|^*|2O^psOObkUx|bLH!cJWMs>j8w4FLsxH!ksv}q)VcPT3pz8#tYcP0a7bzX zS!z2$5L6~T1GGi8VRs#ZAP5};G7gigzUWFIDGvO(s7v-rV300n1k0_?6I7*+HJExEq&oRt-B!pGc>%O6c5W9wI|q zz<^~3MkT$f#11xS!!uWoHXOiPS?gk z4%L=zf;-`jk)YH+U6$kP`DkvMxVmh>KscaJ=En~!Os~l#?)-Dw0%IBN4(Zf>L4@36 zdQGJ0Q?A-75B2Cpo4W`TKbi^S6k;|UME~tb#iF*ltU)vwJvh|v1RDS*tX>sC{>9t6 z#+;o@9H8JKbS#$(9Y9;zfQUxFFYp{pfuhsE&z+$*Bj~+BuP2%+P!O6*Go&*m)K3Wq6L^HKh5OI&*0I`(^w{I-BJ|PDnCvfQN&IT)Uk~w+t zqX7dxQU*C@4R}N*k*jtt6al(>&_{a^m*~0>MCNH+SA!;^dlq%~vB?tMP|%TZ&s6Zi zKnc8F+ESysJb;z_y1WgM?Vx=jh#-pcKXnVxUhq;PIQjsl$c^aioeU``>9L(NLm0xC zGz#`r^kkY%ixdHOW;kMbPhU1M$UWbhwO~oqz5z!CruU26J3I?QT3Wj4X{{YxAYx44 z`nt10XG^8&>94-Fff_*B9klp0Uco4KVJhIC9vSB>3*v{#+NAz<9&T6@Ga2 zz)mp*O35q^GPqK@Qr~9*e2}L#18|`e6WpCZlVSG zb_*xtANXlyg3d;5VVpc(S9!n}!ToTd_BD|M6U6UBy966r^mlvUYEogJD0NksUfN~* zVi^DMW()>4Y=-~?XHfr%W4k8~!^xL^UvQ2JR2hvA_1 z2<(yD4e-O>5VpF^4=L2YI*fB=*jE;GzQbZ7G@)joW5?JS&wOTsafN}+us4?e>|-b7IYHZT^HnR5 zs}Mx`Rdid-6(K)vDUIfiBgbJPXK!p^dXxoF3iARZ`!~QC40uuDc4aGvHPPCGn8=Nx zop9^GKixy7(`a13hknePlbO6C7z1N+VhqN~5yW`IrSaL$AUH5gaO7kz596O>CWFU? zdj76HVxK@wyl?TdqcXPj=$0G|PB2DjPB0DETG06!SoL7$iVfC<8`I4cjWHf;iL zcK-;@ahjL`o8f^YHwFMfgqbej_e9jspJ8u2vnvdyd$T1#*%OJhncVF`0PJCtNT!Vw zpW6clIwrV2l|dZ+s>4-feylw0Pc)GVCxkeS!I28vAfDcp^pv0_PH&KCw=8zgfTL28 z>#aoD2CxT?v;7!B&7pXGjt+)P6Eko`ZVbkc$`To1Zwv-{2Cr<9We6fp)E)$a9V=e* zDmsV7xjg8DSFDHFXM0dTPoe>Wu3IoONDG1peVf}09@rZnjLCp87}$^T%xAk#M-X^G ztwIo?3b49^JqKx%5=j`02NSK`z?e{MU{?pQ#F8UBGZzSALYc_f39a*OT}zuEFhq-B zQACwcC^PiHBljhDhc`?cxGm#p8sH3jV|KtnjG;sj_JFf7vnjsti^HeJEOUG-=^y~_s98zFB>3$U}%n$0D?S%jK&!UU<2C#o}3_U zF;QXmP}?uks~H}*o5Qj@OPxhdl_}tFsIDNLWkSZ-zmB4snD7q!q;?o|n-K zJE0~qPa|E_kMYm03|0W7FH=8(#HGoPBV#2F_ERi~fiK|)zyo_W-uOUf$}kPO>Bpg_ z10X^+U{=FK7NaMz+KCUf`l0uIaJz|1la|R?n3i!F-k8;qsPW7O69m~qMyksKSOTK< zV#!oIMPE$vdp#9lAAN*OH7K%*0>LxF7A#JHb_(-WhWtUVH=auIX3bOwxjwT8E zWZZdTk&2|z8I+?=i_J(8Wk(Lt?sTi2TNZOhfGc}uKgA9X!V#KFbH=S8E64doR77a^ zhQppQwg;#L-Dh(>vw?%aKX*lRCp0>H;6r(!ba8`#CPqx`A5Qk-zyrjXx)sln{H82L z4)TYS@y3_N08BAqa8w}sv0t(c3W5WfQE#qkCqAQRP$ySklwwP^Csl&#HnAl2?fe6s z3O50?H%(UaWe$!8qqq3((YwL@97KGkgFpFV0l? zes5JcfGHC8Y8!5hLTXcWYK8qfY6)E$BNyqifaL?gC!5i)sC4v#vvaI5FhF1;+Zf;& zkKAW)Mhp<@kdwv63l0u$55k#*rO7A5-Yi$9Jmg0ur!th8@}NIPWo?Zd)dlH<%qTsk z>DUCSh(6()*Fe-N3R7f&b5+=ncu=2^u0~b>lQER8XITIvqD4tOK9EVA0msI05P-DM zjj&)qlhOX5+0DB+eiL)SLj|WZL_MvkJRVf6nMtz;`M`;@p{`mDj43A9w(+CR6oEP5 zft}a@&)ib5VS7%+9wE+<`KUMC6+2`_{=q0SbrRC9<@OVHHL1glfH_1wN<#mZiWYr-*_u_04_e8?emgc-P&W`IX*U2Q9SYEHPB;*j_ohIb!pV^+ z2QdcDq$(F}cf*TS4#FN#O!kH$gP3XkNG@dQh#B?6^Lt$NJ@x~Te^?`wT|Tp`DFUZr zKNuKH1)4sCwGWC3nGZqacDX|VdhgyW(K!hEYI(?4*Fv91=Ggd5QY)FV|3ZcB9c3|x zY65R0k)jos3c~irfTHy%3!>3!gV$_q#+46%uv`Q-Xh9V4kRHlSf*RNdAmH+vAg1_y zpc&MvgC3WXjLg;&dp6!m9Egw&z*yRjY)SHpY|ogn&r6gVofaMm^}*!4ARspg*<$sC znT{35aAnVVF{=@ZiN@_|jKtv;Hl{2%FhbE}==}Cc6X0yTK~ljLwZ=Ap(_*tDMPN6W zvk3wN2Vp0UlU;FUOFr_;f=29FPG5k>(BrZ&6o8c^05Gqo)G3>O*Ih$L|4>g7xLVL^mEpn$mKgM*tO>L!U^27-My zpLA0RYuK=0=OV+E5CkEYPW%YrV{d$o6)2XK zkU_X)nbRt~6>Nzq3zHKhBcwT`8172dghBEbt`NbAWCphh>NCnTfvKQ5IHJXL67+fL zePLRCw)!{T5D5}Oa%@o2ywwS3zC>y`V|FqA$%n%3jbR%Hj*J1^MW9&Cary{e=^&^R ztZdX;t&}&;8;XEx>{$_r#BhtNpT^rWi4{O~*pCG+uk!7%H@YQI`dL(4!4yG$aSUw+ zDdvGO-nh=OUdNP-NgerRdwW@Wl?PFbCd#LB%R)EmUQZ;Jh}+6&PZyPsi-_9zqC{DM z35As(rcyK(LZNc9*`SkJZ_!#D3k+Om*E&=LlP3I7H%sG&Ig?1IQYlhlXQR=WZsxem zrbYIoW=+%D2t$B81>k@|;`P#&wYAN$ z{K!o$Arc+}Y$!csuvRHYTC^^teIAf)$GhGbp~ygUH2Z-p1v+qbm_8BSSp4I7rYwN` zmjzJk>zktbW-AMOL!F?=7)R(uxZSe*Ktpg5Ol>3g$S_v4=t2k%53omdBcCv!6!3@+ ziJqV|_*7A*27oKzUVLxgcJtxnR$PdCeDz+HLn<;{$BXMx!%@i5-7%5m#qAQxg zZZ8WkNn>%1Mn~^OksW>^Z*0P@g6-OM=*VUM1S?uRBe{((JaEL#J9Pi+%D3WXoj z+1Nk(EL~(^kI|me<<_&Fj|$mobGMdmEk!{=^R`8Fu)2P_Q)5yx3;oXKFo+V z=*7(#ZkZ@xei8DEj5y06Yt&Z+NEx9rhyaL>TsJ|qYD6&Iw)F{>Cz{v!-T z584rD+-4&c6pB3nPRoJ4$cV5!L|{gLU2YIQPuWtB zin5>=S)%>xsV&IAX&uH<;vDbxAp=lw$RbpRs^h{X1)K_pLgSJ$E>2Hb@>B0o7Q{GG zQ>xPmo-wfKR=h0mZ3BHO(;lt4EQr0xrU(V=oPEMvtjNY2Mm}oE<8s%wL^&!ukS8c9 z`UCxP$OfW8D9D5QLLV4JwCEU!6~)K=Sgb@QGn_8O2$n-gF&a;*nxk;=0bi6)^}e}! z$&O|XE-wP2Tq+LuqjrQfPbdhENDvE|@+!F?><7&n@_8`|Ad7Hmg4jy|Kak#yQOFZr zEsmIR1>pb#f8!cL31XQx7D#6>0YN}QKylyTf*^9!>+-W|PugHSpn<5Lr!y!%B1Ny` z%41j58`evx9P@Q(Cgk@RdFzPZUS!jMAV0C-LAGhhQ7Drz_6SqAhnVG}fqZYq1`-Ft zkf$?hBM71l{a%448T@yVY77&ZS`_8T5}jZwaJU{^bivN$Lmq@b(85i>5ZbPssF>zkwb{fC5u-veuOZ+0@E7h$<> z1{7nGh7+U;B1gAuXIW6I7*pjzKh~M(O(Z|q#bu#ic+{(G5JV(|)S`%jT2|6_699X3 zA&ebf5g}(d)MTqYAT7r%V|tyv$QX3E;`u8d_)*IYT5O{OXj_FT3!n#7^!q}Z?IF0( z6H!l)jI;)@tlC1K<2Ds#x>=iJ0npMS=vQT;=V7=lLYuG?fHiW06&h{^5!&RC?u`79 zw{#cGHu>0#1J78%C_z_X!BcpxAPB;LIi0A0p7rWiygFgaf?#5RXogIL)Q5EAH5s8? zAvz%tF*(5-{6N-Y0^pzNm6l(au+>9adRwG*KC!_mAo^@lPBG0mycP7*eh?h1OssLD zi6m|nOwt8Gd8@(!%*QCKwkm>&r&{TKcxlTv224q)XUrn&3e99pz~{!oBA#mdtGita z77bWKQ!T}lE%d8tZErwl=vx(Dr#=_0t+HT6g2@x>_52!Olm|CffyF#Hz}fvk0~L#A<&J^>_=|tYA=?174r?(0lB7R*(Xf?0i2V}q0<^97O$E$%3QeNCB<$(1R9J%CfXBQl^4$x1k$^x#8tFDFie%7|RKw{t)4oOo&xl!7SgQ_DvwOWg)ggKU zBoJh2i53U&Tpgxglb`}i`!qnvpb;@R`5xE-YOhyC@cRbHzrq!w+QN(bF*GYKInTgR zsCE{)!yr1;62F&3cTL%B*nzx7M}@R;)C}E*1t_5Xp^H7A@IYG8Bz+3T;bbm>vB7wO zf9}SVSYBS>k8V=QK!G?MsR?kmga{F3nl>g(?a*=6(s2K8VKH&BA ztO(ZBH5cnI_WFVRgfMT4BwTI_BLz2m?V0l04{8Nn4pIoEMzr>pf)i^)?7A3$G2XW_ zgc~r>@NDp*LYLzZgX*jH*bfRrS6M^3cLBJgwHYR>Y%=llfUzKgu z+?7t0p4q6UM_4K3=9)lN_o@h?=^?4|90|mYCR6hfDWp1v$!CcS7b~)Xbo2oqoM_2Y zsQ)MhmJ8E5oVAq#gA0{L0hNL^AJiu%K6rEJJqv<(Sa6Xd3GG#QK}5p}*+&8BuPO)r zK%Of{LOnE5A21_2B8Wf*mM?BMx{Z2^w=l?PkFo%YR_q?cQ9r#agK;(95D<}Lq|k3S z)zX4XLfOr0opypiSPo(BfV#nGwO1aZ!>GsN2fdL;0fU5kmItsJhjj_D!MOqzK|fZ2 z5N7#Pk^w^qw-vSOzDQdY0yTp^g(^nhLsp|o#Y_QIVpM)r3y?xSdOoc`O@ zAtw+-7!hx%pkec+(7aX28cph3&xU4 zoTV`LSC<8_JYq4+vR18nb+kA`rXmQYLx^uUkv;ipMR_MXQYon$b?r@^$-rmz8(5gx zkFv{w{o_A^+Yb|5{pEOQBU;>XA!D>?woCmi?4M&O57H}3Sk;j|yK*tJW&`X1W)7@k zc`}AKvbTA|yc_V-J9iubpB-SfLs{u%3P)LTgYX}m$ti2j$Ap46GAK)i%>`msZ*WwX z(d`tv1XjJMq}Eij-ss`5S~%p#tQwQB*MTQHwite#oZs!D+DvGEh(r z%xD@c>EKb_lk0fMNuAjbK%+nkk&1zS6lRZuE*T6QM5Q!hSTNBW_bj-OX?3ei1)=7n z-9kuX*cgeUP$(lpsrR@kT2l&ha2kjT4U3yD*uaX<1!pH6jYe6M z6?5#V8z`G_8gA`_Rp^NgT0ync5lXj_NpoiKn|(r zom6=CPrRxs5kDb7!{ah=t(E_XPR3vH6XX}0B>j z^;d)8scwfz+`)mG#u0ATRvde)}#D9~P{*!eAS_-L8{ICrZPKQjIxoJ$acF=rP z;Q$0kV}u@x!xxpvRhA0kqvYwRpqt}WK+Q()?C69ReIKg1NM*q9D_O7t9FGAr=o-jZ0MqR zY{u32;XgQIc~&0qqp^+lne21nBEcRx#|=7Fh3b27vJV**GzVDGGJ=0!TxiHZvc*&M zKmyeQD^7(2G6;pChu~z+hGK8V4`8!BW@VvK)?|f~&H&r1Gf>X9%FGsuoUm+DFHYj- zAD3z)J7?3t9+^Z>Rz1X)88)bo=|LK(D-Ap03~z`Ky&p-dhpIaBCT5cp2ICZh2;B^$ zNmUEnbG)%3L(oo5Xe{(z5=82;hdJ1iYF_1wlj9VDNs?oOJsbGUAP+bY14ISe=6i#I zClO>DR;__;Abki;^)qzJfX_|39!N{vXwZ|1L<)tWQAM52RTc90rA=l-SflFj+2rt_ zV=xCNXU`1PNh%#r&?|&ya%7N0jQY6h+qmQyV5kWAkVV|`s8`H&DmrF$W?+XrLIPms zg+M_396S6NZ)`wt#8uZqKd#S*{lkyC!wTYz%f=`P8O}5Ek26FN&KNn!EsBHr5sRWG z^$ZUZ47X-jGEi(>Gug7xe+60}l}y7~Ng~)}%cQ|2ku|gyIqQuLtwV-<<;1{fZz=}+ zK~Bm80~-kNiV$rqq8_YjL1c&h;|$LjYc$-@lxpuXP6cNjb8b#>2@t>EM{gam1=|}B z5L4)8NLOnQ*&!En4|D^&9OHvMn;>k#PUbQnwx!^?-t?t%VuN4|o7{ zChOJ=uz{Um!1$&6YxTgAOU)ny5N<4-;nN&yaY2u|VGnN{DeO(O%7TCyLI86s7N3x! zkig_E>!kjwc4t#Zs9QewmUAZ_JR}qh@-g(Rd0nC-j#`D^Xzvs@St&Y!=z>d;!|iT( zQ(tccQT!8xO#{6nrHxuhS%hxAXsMHd z@WqU%*UdC}kp7JVsWU*Fh)?}$Ih(qQ(3ly!6DLY%1q^J4oAC+AMx6=epY4%JE&%75 z9J5m+tJ;wlDk)`OXVL-thbv|xO!=62Pz18irmkaFW)(miaKY2ggu%6VfGzxXl7oavF+kt4pz)~+0uVc_VaDB#3QYkJ71yUd6@*0x zK$hTyenR?%-1>6`h1$3Q@k09G?N%DF5g`s87BtgEQ=Pw0Uq2IR;J?C@}VOxVJdw4@Cu%Z9z zY|9LJYt4MN&tt4XP@_%Zgax|Vr$V;{d{OXh{X^cOXCa@;kz%H(l$rV-?MR?6fsR2% zLbxW?r{15wS6I=95rAQ)6T^w6xC^+CEE7(9@$iJ$nEbO1cvq445 z1~$W<49XAA?1b41)uM|`8PVv>XDbtDIHKCSYwVk_g-^aDo7B~ zKyavn@%qQ(u$WJeix$~AX}d7VAt_s-Q>Dc?e4~C1=7nWbnEcHau6>zx3*#{(Q6}z3 zdJW}y>kfnYdFzu=Pf<+(%^fp?G|6p^LEU^dPiF!m9!g3fluGpTX5ZqDbgZc$NQSuqHC;fs>2xrKT%doB1ncNFG+Z&H=%Zq zfZH40FXT~VTFpXf^2Tug%|}6P#+nDH+r*)SIS%z*(~r$Wuy7Tu zHI$!l;!S1xk^!S!K6~BeLGAO&N`o;`;u#b$si<%M&hEEb{`I8!t{U=E@okDf=qX(CKnHf@Ghm#A8#WlWM6h>%XLX|jlW`h}cx zi3-jE)jCS*E;;>m2O&m?LV#agfRd8Hb97kRKf(VGsJcR_p*F*7PhT&xo){v9yL4=x z85Lem-uB|;!JMph3}HM$08E3)AV%(Y`T`m%A&g%Fzhb^ak?|je0*pEQ%;R_d;F&T{ zu%&|s8T3)F$o-)jpLd&Mg$9pzHD-h2s}X-Tguv{_B@T_^gt#9qM-efTECsV)!rL35 zqUzRK!Fdqb)r;5hF68~KvZQd^faK|HUn0~KebFH+`MtIHNK1v%VFL53JZpK>S!$1HQ4 z5(zJnN}UnvSf;$R5)x~R7|Mo}kLzh~^nj2qoB3W-!Dw;S2#W;H5DX)_y^o5=mlM z{$mWuQuh%s%Av!U;OVkhzMz? zHe>e*7Z;iT%#S(G!yP@SA;ae}cX?VIB9Fvpb@)+fG6ZVKgjI^qyE6{^RvS(0A4M(t zmd0GLj*@Zv*wj-iidADXo&s^8M&9u#{qd5d%ga?{-I6+YLzltkyQ@4i;HuM8p31%# z;{KxTe)8zQEOZ#+w2g`Pz}NBB6y|H9J%hkwr(c>)=@!G?v?50xHc|CD*k^Ry#Z2KP z{gj``q-Qf5pk-Lvlk&?@Kf)pbTe5HE_hV4PbsH=JoLXF+fw2gS=g1Q5 zUdit&)EPC=ph%*Wd#QgbiXLCo<+~B|UB!9ZP$Xs-O>v=kcqsmliC?Xh6P3A$^^@6V zG%8VyT#=vpnfR>}D%&aht$#6d?*u6Q2CvJ@q6{dj^ny}b+L^}!y(4L4giC}}q&!DN z!FW<202}D(?BMmO9`ev5H(vPqIpj5SG+BXx3}%}Wc@lXhcSwo(O+dB@a|mG^8w*sM z{Pu<8jyA$;^2-!j<*>O~*p5ja)vhog*9#^RTbgYX`YhmC&`~cTPsQ`7;ODzP=d0ax zdJ%+Y#wAX&6oRG_pNf^%lNP2(pfr(uq;cu#SzWI>6Kqacm5w-g453Ij;$SQU+vmA! zbO?Wf4I^dcu?yE$BmqpuOCT zv(l4I#T=*tM3_7nB7bL4=A+}U^ zWxCUqwmF=BLq!_2uL3ufrXHk1N}j6z?pBjJV<5O0`IIM`RO}; z{z&k9pXOwRxOzG5T z$z3@$HU#^2u*kZ)oDRCa8zd=RtbOgO3+kHFm8L{LBU+IHm}+Y^%0 z-o=|pw-`+){z#f*&f9u7%giMR@!(~c7w;HBS~F&hjJ#$9EjtMD?vU4aLswQCGLt_i zF!|M8$oWNCift5+$l_)dzA!jS0bl=kBy)!&AFL;IIjbGyGpTXsln_}r`avOkmqnON zK-DQiLw$*G-5Kj0vKzG;q_1vUpPWarr$9lC>c4{!`vq){tj#^VEcrNo1dIyw3De_c zM-Ki>BnXZZ!2kA8#OM=aCENEX%*RNzh1S@S?jqYcV9r{sVRg|HuCf zlE0EAbCVD}7BdK=*U&m5lbF+_oUoHk+6NUu%4DZT55-ge31JbjPelI-kxbf5?aP#5 z!sfx47a&WywFo`}yKsMxyQplv} zS$DWL!0AbpjSWe{W(L=5<{iw52C@deQXbz@%ZB(KM7tuw37SWvX>RNq#oI?Y7lXYMBH%< zPbIJinVDi1hQaOvLwXK{?T-vIo33#x!A*bRWFYq7ts%T8w4l{GAkzfRLL1uQMwFI- z0||Yr344R84ZGx4q`Aqiwqu~b7WSdjz*96P>wlAv6&@u14rdOFTkiho@786|U+Q?; zw@cS6;1N{yiKq;zjSOaL^qvgkQ4|&X&TM0z*c!(9X8hduHm(>xGn1)WOVirl%8C-y zQ320s_!4r8lB7;_q59b$R`@{n^)P)d4_U{1B^77K)BJ_bWD2oM7<1#}sT7S%7X0mh zrF;V@vRfGc9=W^-o_LxDlMrDVl%=oHYYT={gziNVmSV4)>}t-DXhTjzepO0>{Dw%R z%uRe5U)$G|EI<#{`m*Yj6igX9laEX5;ab|3(n$lKBm0%bdJ3vm!&&`qjPvDq~LY)`Z;gc6#TR1>C8k{;!Ob`D)V|{;db|QLt(L}cx)z%u$?XgL~1tgmeNipBzVft)@GEgVN`)|U2w3*BAJZ2H!W}*N}ce} zaoUXW(Gyb5w55rg;$?|&M{G^C0Q1?&{ z(+vwBW^mMl%mo`n&QE&+Hi+|#EkWe*fzJDLUUWD(G&N3Y>AD9tq3gUd5J!s0nNC@F z@OwW}oPZ!4lJw6RJ`lTFXx=)1iIGLe+zH6U#`vm5aK|XT`-@=FNcl(KAR$x6r{RG1 zs3({!vG9sjJUuz3kXMv%rOHFlOGa&=rpx|aKW##a2jsY|a?s{bAJW#K1ftiD6$Nrjp?w?}7&fd5A+9P-{S?Ppvr(JqTA{ ztv^t6P!3Ox1vT{YxG`FS1^XF74V=fS+aR1u)4JIIpeU#|P_azfjpSjZoI&WX*))5K z{zRn~hRRSWHrBq2{RHpu`5ct;RA_A4ef>3r-t+phN?vgWU0hTb(@x)`pTq-(S(c>E zEE$5QZ_w@UKExAA55tq!GX0kK>|b3!W-JhW>3SDCt%z4WY#6igGF8u}Mw4f0ol1$I z?0?Fof#cqJ4SvJj46ZUeDCX&bT!Ifz5J|mm)Ba73fMlpZEiRGbc|@4gc=prAUWuaS zz2XNYyk710cpfi)Bd97RHoDb}OPIO=@wN;JD-eG##-SS~jgTPB6F&7&wUfc*+U9$M;;XOu5l1o0$)&6a}}3 z{DN=ZhK++835p3R!mn{z2AsLpo{0eLRRDWBBf9wv{2qJplmpw) zuatczx(CD5&`hy{WIi@Ms+A0M1T6M5oV`JH(zucN!+mW;uJU& zeEk$93TM+64cErFj5>#_o||}h&z-k3%5OGbm;~AS6M816sv$5@NH8Y5$`xB0#p0I} zlqJk9UawX9p&asuJYKtm%jh4xR_y+SJOj{^NAemr)!5XO_bvYDoH^@ zKe2(botU@NW8ZVPzvHbJr83{fy1I!O8>n}(>X9c%keI)%(9H!Dk=+|_vp^dn6wq{@ z4+jU}^vt&l3Bm^*9z0`m*|iIF;y|ZQ)y;Z>D+#pRFP$S=djI*QeW|-nP$%7GQ@4i*Q-2@*7fk_j8xO$duS+G%l5KnLdA!1OhuY zo)T>MwKFYBHEX3Rjb~q;dOivDf~Djn{s(-rtxhD51WJarF&$&Ea%r_XkIK znB~NCpt3RaccQnjxH9(+t#=@=(}K3{QbXzP7cG8@6c|@p}6$na0XxtYS^>xbcP^{MDPXDg^%lz zNme&FGnb&--QO%-c zj5~_%5wp@=5TwFOk*4P+440&N^mAt{77%AZ8Vz~yBO9EqDV84_1oJ-Gy16SP@;k$( zZ0Ld7?oi^f=BT0H6fUx^e+ZmWQ;I-Whe`@Scj2UTec#BnD2T1+%bp0bLrg)(hq_&4 zLOrjoNG-;JA~=+b-i|?eVedlhhl&dLp(hYBrDJ6-wAINW`XvT*u(`g9upbxq?;+PU z+~2kgm52sCdc=b0xmoQwlBq{cv5sE@P46#$hH~?8Wyaz@iL+$({uBhCS{-cu!m%%a z@UxF5K8C3}C%m})hG8BFVtiljQ)oON3cH#2UU^$=H)FPm0;$fm;90JSMV+wjvqb^2 z5T+TH5%wA~VmML=Y4{gQ^gu2M>`Co8GTU2EBqyFcZ#F)BK|E~F9mFlhl-#gM{wgzi zHIT!#z>Vt)+74t)+SiN3IhbCHtio;VR!7M(nfz1a(thW+8w@?rm4nv_Wk{Qu%`S1H z$iBd^DbSUWIvq@9j62X}392rcI*<=h-(`o8!(4=ohhEc_g3Q7j*O^U+>>=-B6axJz zeX`rk7(M;7{C#_*&DdjmWWC@*gGmu7Dt({r7z&XpWdu{bQ6#tzC{IyYW#5?-L(KwR z9mGQ-WN3>dZIv9bm6L3=@5s2Ac9r7G&<{F?$i5?0OJ;#QSM-NVVklW5K=ERzo6nvN z3St?1=M(Yg*ve2ZTeaWWeiR&t%5~Y!k@ze)%K==<^<-0yasZRP5dQm~#mbz`7(pcp zbTMMozHRXpst>jFc*r)3BI1nFwg3bpmFYmXvmC4r6g&(A-Agbx;bI~kxpKESAHAqS z2Liv^v^nMhWX_q?tEp9qz*+$Kv2GUQ-9C;iUtMd#MoiTXxonfe^^dn{x(=n5*K_Or z^qbJ58vex#UqS+hrLYG;>c`cruobS92$uAk`)R8qCLND&=P6o4TJ4Ectdw=d*9Dqn z`?W~*i3#=E;c6E=&Z9`&!VKPO$tHB!1wl_q`G&*s;+glW-Bx6a=QFPR!;M$_V|A(o z)*7;N4s$wB{qpQn3=i^VGV+_HB;Ns_%(3rGzT55x>AdZ>BZH zqCCv2Nmiz!4^~}nBGqrOXez3p&bZC50W;%E4XJqZw5j%UPURF--T4hN{E6M1H5!8} zxhH{n6(*)80Y288>NT60{1eHnw&!U;TBYkS#doixUaR+-Jzk*2eUxGWFORMJSavLV zmgh>;?@)P{qNBoU&AzJ8f(|kK?_VD+CPy}zOw||b$qrDXY5k0+fa}}$ z`<6C5emd>o&@sMCY9l+>G&J^f0w2_tt3KCO2G{d)b}m%9>1^C_;GzgUqp38L*Zh`} z{@ncv(G+i;A6uLEB9Yow)93d0#X47_Q?-^DDTVC!HrGJ>wI+ogu2JeVu{(4 zDz*(c032QLWw3mHVpjdI*bnd|Pn_JRZ}w@K0@`YNPJCAI{mtYtPme{*GhoHnb~`^s zQW!ekew>~M1u2goJ)z~%x#pTNIibOR(b`5f%5Aq?I}@s^K9#y)0=UU+dNdf8kWqc;13ltVm{wD={9goiQzapDPu8xb9yn zX?-h@dD9a-3p@C=CgZ8b5`yvOJff>neZQl=Gp^%3)T1c&P+xkTlXou`FVv8ShB1ju z^Z3a$QPGEx*CVRr0x|W#8OrCA)#*rJW}r(s64a}=r}DEIiRculdEx(WmSM%j6ee=L z;*2=u_BxgjRm{@1GF~xc`9M?YTAZ~q&AlvH^_$3y9Xpx3$Peo7Vw7WN*YTkI7<4lI z=H09BPn8Q~xrg9lv+Gl-uCa3oAfkZ4XRs9wi;l;$(Ff6Gu{JnDgMXvp_Fg`Zj^J*0 zS##jnY!+(lpn1T zrO+?a0-yUPFI1D6(A#G?CeMcV`KoRvdRR4pdUj-!wV|TTlE?F#_Zz>gTH~dcc2PdN z4f}DpF8xm#YXG~sx(;yvN+M7uayg>&zB6+Ub-lU*cyJ2dc6*iL&-i0KXQW!>auJwCITC zZJjZGUDgy|puRYOz;Z14MZv{##~FTq-4O8HxA;1SWAV=HC^q(IrO`WoioAR#6(k$$@C&b zl!wRTAzCPXoAr3WX@hmi^M$!OJ)PTHJ!7o`q3RvM?1aZo%7O|tb+8Kc!K93Bi2gvm zMOIhLRvxY8xP@m~B>zoN={*j@vga*>rG#}&JNo+y%;TY^v`^g`l<#^kfPdiISq&G4 z$zi7*!zR|*|Jp=@klDST?;0)HG0~umTp|2&TKM*Sv;$4h5Yeu!Aop^*3oNF9 zS(8|r{3@jh!he~HgI@CdvY}=Ho)m)S8n4zV2>7EXTq0% z*TMRc!dkNtkJgXwM&fo?!a2W{7lHvg2+MnfnX5->VUgKTs@Z>EDT)C{;1Fm$rmf6W zU7u2Upf`|-_HW;^+(ppAIM)#8=b)*2Vq|WM>%&y@8=aZ2hk~FMem+)Xvpvtq_`c_n zPa`LH?^G(6F%rz<1kIbXM1l92PCcihtfGAV>oT@%xvWwLqpf@Ml!ejwuCqkK$m2_M zDcJSi<2(NAZ}v^EBT>q$@R}4cB8ejwk&^=%wHRmhIdv1?L#m-v#9OrKDpu;ev*b2( z5{8BT4{-}Frr#c^H)&^~Ae8p&3yc2bGh9KBVi!AV8Lqr-)NTsBmMDzs)#N?h#hF;; z&e_i_W=^u;w)~)EKA7PR-?Ww7nyN>=5+@pTNzV9s$CHiO>l zV@I@MUNbKlwO-4|uONHKJsAIC4)eCcZ|flsoSOHV;oo99J)UQym^kbRoFCDRxv{K*Z)dN{^^Wb)PUg`wkOq_KWl@3GA|C4}ef6_!b;R;Z-SeVtE=+ zauBnjsEt{K;5pYm@qb7TrrV_d;6^35ji6zq%!#{n$UO4yC#~f zP#??<95}PSY>56}b%`Q|Mo5Xs4LZ^HxSSrhM86Dwgk{aBMy0s@V!g@wD6UeWP3HAI z-MtESwJsPupwqHhvF@I%ayna-`~rhigGu@l?I6bLaXT*RVlhvnn|4!haXeRHp_s_7 zQZYpaxXtT}*6e|Lpidr}qaRB(1*D$K1x#c;d@Qw~`4yGCA#b{}VWVI)bh6;A_>BtC z=4&N08IS7dwPPjc*Cw0gA9_-%VzkT*Z)j~x|2T4;!Gw>gl`nm$fZYAC#kE4ztbmzwaBTFy)}Qpcqf^on#bmR-7QoLUHP@WO zfAV!!CVR(az@7b3RS-U=PQLn9mVSPA5gqFxb zADX!T@v0H-;N-KmS}fx3HNds1Jxt!ix%YO$ZQN*{=+?zOo}dk{IG5B)`?|INOPdw1 z!SG_rGSpLsa})besD7HS9ilyqkJumDWVOd~JgpyPL?A9yk=M~t%m@#0dK zY~|O+`Fho*8=!@46i9|?n%ItD2y0FWz}HGIlBZ%+N>EXPRsLr!z_RNOXJxe2jjNIN z)nsMh+&$|`q&%G)kd9p}l=G*&wj|x{)$@tDmTkZ`PDTKT2wpr#FEyqS^od`uYi_gX zeVLgwd@ojI4Ff!{J5nBQFQ_-Jo<|$k4z!kA67~=Fca8E%X`;uEQ86b2F|*d*Y=qvE zaeO-#j%8eU@(XHCS5i)}GX3yt@rg*gs*sJQ zY-z%w&m0>X|DjV))Ujbl*l=>X(C<9nU1O`Otf6GIT3$15V`Kcp*~F>NPga_qwq&m5 zXNQ-rxn!GF`cAQOBop--`L1{L_TWZkdsgzSFLAnjJ58sZAZl9O9^q~rsNrhs5n_7R zm9=)aZgGCK&b|Ir{a9Zf^16CjyH^o3vF38IifI3VLZ=M~$+(($HC>3X_4(7XW}|;_ zUTtXqA#wK9AOYUQ$XDT$?_|Fx%lR_!xQIOM{r)s{>vImN8;tzs>O!viPPJ!5Iw0b! zbi3N}fm4=at^KywVLz0)Ws8*4Q}pm>}hLbIc)_JsF5A9|*m`)T7sBmL5S)PF{s z+c|s9xS;h)I^=o_0-CvQqKz}CV~~}QreOEbH;K$~f#8Z+JgHok&FL-&@SeOjb|I5? z0A34)f$@qAejQBD^mS}z=J$Tbt9-Ys17`OG4HUqUl4#U@mVpw(K{!9`F1fwXlv{Pd z<<;O{a5OFrj|?LnpyQ5VVh+U?p*yIW=BF)(?Wacqi_7!p;#M$L7E*=ivl3V!MM zUIBAICZWwks>E)h;uSK*&&B$;3gdBdmB(aye2;`8i}+4=fKy%o*|=Ic{e2irySSDX zHUAk$M`WbdKm%dCnGhP2!j98w;-Q|;PK9sc8Rw!dc1Nz@OWV47BVy_sz+~HKtjbtF z&&(2yct{Q zN_gNrZ(T+LVOSzrQXCsKtvn?zzPQ8Xs1#?qp@MFc*f|iN)+o=!ly?4Nhb6FwH+n!8 zOXlhBPPAEhNtb701F|wNTE)^xRf2sl&}r=SwiFTxYQDTCRs1ku_vrnIIly@fzNX0~ zor^)TgRFX{>){A4bV8a<`iIV|bQCv|j&5{ia#ZXa6Y;=jK;+OqUMB7*`{VJgM^sa^{>%YWBiz_QC`kJMg)f~%3k+0Hp`WY0LEKeswO?z$wL3XPv7R2iTi zg;!ha%}Ce7geF54!gUynI2o?h8yYH-}~7)V(7G9-o3-qWag<*W;|)4IwXiW=7&Xd+%THz$F`ZY zz!HJ^3D4dx0=z80kCtOp|EP_J{AaYEiYVFdCJxFq?F@4pmXr~yD5$(Cv+cZzZiV&$ z-p|?K*~Vc}Q5Y{mP15^+;;GRM5-Xj-{~4Q|V5bONO;;8=Y#P>MQAr(r&Bjx)H?7hE zzz9HeM)fQcrLIVmZJBa3)m_QQtr5mw272^a8>KHlCqQ|)IZmPvBx?Gg-*jYN$1A^mEo|+^MTC#@&VN{ zl6;tB$KSvZ!Du7CeoYM`I(iDb-0+@b`cku`U@QXm^`G(C&iXmHsni$_rup5$!D$)s z<5Z9U8A%@l^SW@%D}6JcRwR3CFuz117Jz|Wiky(vzxM^R4wXK+a&qyXtoW(Jr_}wx@1LH!9on0gZH)_Hc*F)C5 z9|Y-ZmWzrB8HY)^HwPgE0jAdCum?BB%*?D_Z~yxCX8!K`0&ix67FZsoMdzq6m;m>* z*|Oq%v<#s6cPl-@*CbN`jIpTTqgBxffK=~2O76koMkOv(4d*iFZ5h0;31>WeGByTh ziSlk&LdO%LG*Dyv_q{|pK>7JG^`jzUFoB!mchpikho;dv^`rIirof2bzZLxH`~Q7j zH%v$6p_FQ7&vHCjO8%M|hrOU~C;7gF5+h?O$1kp|h1aiLacGpPjy^>cJo=1Zd+Rh1 zBDiasu;;4`GzaY^^Lu`<{=JEfT~IAUk}C4Bprd3<5jsd;D8acW#CEWj;LrWV3&s<53E<`!H3w ztS)a`VfAKA>1pk-LP>nuZo0e);|RJ%$}ph7X=kU+HbKARsUb%=hs2(sNA&mT0vUq6O)uBZ?CEhS2L>BH58xj zh}EQN>mJ&&i+>*9bib*D&mkjImrxCAq4teEopl$VW@yct1zmlV%wBUW{LUuPK3xcg zaYh6dbB^d$A#tArga0|Cb^0g4B$!>E+)NKye!#_xG-(J`-qdUWJ01-U&FqoW?BUbP z3n9Oc7L9e_e1+~r!`i~m&JMH5XmVp?V|sde$7vLv>&5z5dH?IvO<(T3IM-maIZNtP zp7=}{91tH=noIho-g-dhqoHF*k90Ae(_3>(V zcNd4*WYu-U`=aeZ$K9RzW3`~N(ylKYqvP#9ExDt3)a0gIyM^!Nl$r6vG(IiuWzBVC zEQ7njcB8$b{rNawDyjD$Cb%?JO|L1ezHBVbl{2B)DH)~}BRQDX-tviV6}G+ELsKUI zsJL!0s@V!}eLD-cd2lot*OIp=W}+BP2rn@xzjac7j~%9V->%mnh zC}O&@C#b>%6Bt2(7=HDgCAjLD;HJa~k<6Q#%79YG_c1cFGn)P*!EJGM*09n3cE95N za-VNON=j3Y8EIKj*AbU0aKIuR~v zt@z`|4-1QuN0j+m=ac!xMKvyQ700LRz1w7+Hjf13{}Q&w;2hk}&c~De@iBx!yTg#t3|DYyW!T(5~S!S^%#gckWu) z6PJXzF*E0?1ANo#OZo6fp^_G)smE`x7JWQC zJRwN?v|6Ra$JU+n7+rv%1`-qT_ z_G7ZRc%9lZN-_U|cm0hkqhgD>psVjoh5p%BRJ)(pK7~&=4Cc#B)ONGo8SFGr+Fti& za|tz6j-0CQa|-Gf<&6f-sir&)h1nEADc7Jfqe{T7 z7^aGlf|pU!;-t3$K3(!60LdAt<7+g(On*;;@91UCaY%u;BUI=)#b zP>_+Jprg};+%9O1mNr8yB=yvEi5qp?$ikx5;&Ns(p3bS?{aNy0fB*RIx#LuC(idrb zd>oBj#_MiQl~P+lq1t#P(S%U_pWSgxa@cV_A%%K` zHpD=2E3AZnxiKJNkQGF{7>qu8@|?_a!Tsu+bI`O>f{|jrVPqpg2C9DsRcEsuF37;zxk25ac(b4fsh1E&C-vHK531>Ri;`I66E#2TiQVGtu zF7=c^TfL57#%i%O9zlxnwttwfy^C?>X^I=$kD8M|4P_2*MG3egObX=GRx|1Na?&NS zI8}XpHFB?@!3A|vQ$x7h8UJQhCQUZT^3 zk>Ne6dINOmU{BIzN(-{+hpQ(^SGRnlVB|YM$<{XlL6>EqJVRXo$(;A!%nI8FV0aQ$H_o>szrZ=KkZ zx;svmGEJQyi_w8X` zbahQZ>ICnDmWD=Zk0((ztzzBJpFbVC7Ib~zW))?fkEVmmq>{WZ*4vzlw&4QqP8MdS ziG)KcPgk29DWwies!=^!tt3mQmiBgcX|x(A_rG;~yuXBp!&Sv4^y3@VP}9=pTS|q= zWN=lLPewYqu_O^V;hsK;9S;8#z3a?;`b`EfKxWsbZZ;tFx&-$`S_naMm@85CeOcL9 zyvyOl$JVxP-hVWJ{j18dt;06naZie<0eP6`MiFq~h_$yM9Qm}ZQc5{WDu%t$ekKXo z7z*ffo{iNiWAJIb^R0!8AZc(^X@8jYev!dE=koDaethX%StO{C;;JWFUn98bm@}D- z+;Jz>Ug7qH+X2kzw7)G>?oF&sV9uS*)NXW)@L=&|L+6{P%?}6%V@Xk6M<6=Yc4X0e zzYKX6$7FmuiwZ=)%Z!5|A(FW5!j{+=hXM{Mg0932`{ zkXO4KV4LadNgOuN|As*HqfAak9UTMHG=HTG?f?lyYqFvV@lu7nyvsCXbH7aD> z+@7P!jKU!Z>dj6^TRoxb`W_z7Ef?*cAXB$q4^tz(w&L z{*U)UF{5@p`sp^W_cVJudZsT%;bE$UHB3a5g88r zdOHSIEGp+aA64_;P743kDsw;O!2^l7u1?`W#LYfrjHc z#@rvm);J!>U_RwT*Mg=gZQc7Z@hZ#r6y!Z7un=Cp*ztZ>;d#;KI+n_Mzd|%Aii139bFv(S@@H;t-iXD z`E%a44Z6lrgF0zz=kNJYI~NC1-ZpKR%xyD0)|>d+e=e);a+}>vhz2gBLV>UV1L5VW8F$$L;&}`dGp~y&*yh+is{5bq*d!pc#cYm5$X^riZQ4NobM8GA{>%PC#z6_$_2`1q8e|6ACJ(Q&3at{f3YzhVH}Ep(_KGW4I-Odnx~hIr!2tS zR(CKwea9=4b<}v3m*31Z2~ zx_t2*?G~5P*5Jt=bOHjKkI_PEv4toC1C8e{Xu`$b>?3nodPEIL(6w^j%_7qpPGcM1 z4{M1nBL9J!U0@YsQvM)KKD-olpD{qu^|Vp_@y>xCm3jDJ9tp&0e*zl_+LuBA=gaz8ye`{7x(4iuWT_8^q+Pr8FvaWOR}m=;uQ7+b)<2SG^KUvyGTT{a?D&7>ado^yiX;(x z3T*-zbPqbmjdqqXhH}AjJ828$_07!&Xdji9(nx|rp%oBRGFUC=FSmNcqY2*LUoJrI zAWf726ji)~wzMYln%(>BlV}9y-u`~Q?S{|MbiUHol4;hQMV(@S^b_cMLqkJ)?dD## zl~WM7KpskRfYWA`V_q}S<@n-aB9Tey02Eo}=jGo1k@$Uou_3$k<&)%}!or!U>D;}M zB&jMPQ_}-KKffH|P!BgZx%s)dJJ6(s%wL&|ezIWp`hL9VHfzOWvzSfZpRFuZ=<=J4 zVjqR9gW-1vEj?*Z*RKmyywat=ZWM>n}C%5|>sdC1EKyUbMfc z-PfsS|yPb@T}fhK4?7;%G!4hU^t8}pprSl4L!BBdG$m~NcA z=V=Y4;NH+yT52!SQnyjElO&H&Fjgs&hBJ+5u33Gf6HRJ2DZEz_LW=<48kdY1SEDLI zOK}522_BlAZqXZ@YwAkCoWx2s@Dq(zGA1?V9Wn}JMI@tQHM7|kIx@#94Gawl-8R9wuKxApIqs&S8HznFC5)xEdSZ-2T^< z`K1lctOf5si7r6dO22VlWU6m04cL`I>+)r)GF|?nbMZsB@r9UYWc$g>rt$L*jAS}RfrvI2j%Gy0ANAf;JSl=s1HZ@5 z>1glZcxqwSW#LJoFqg?aU%hK1O0Jw~Bx4k=ZmH>a-yY`g?!NDl<%1bFw!SJD@qnkgui$4UBd2bjRe#ooG?IEMjtDEHR-k$f%SpylGuj>L(wc>oa zE&>%KTV(m)Y5|u?9NgTp$qZZNE$65=wAs2HJ~syw{9ccjBlI0s_*A3`2?=TJwsc>h z_zHUTcvGz>-Gf6y6o^Sk3@0*qODk<&Kr+7G8kaO6Fc9L?C$${bW~5qx%e-cTjWb8* z?M^hmjbrT}P>#lmws{cXg)`-|1olmq_bxs<8kxFQ3|ayi=R~Z{cmy-lJ4N&P8~*kE zeI}ML>0^dPOty*X(42_&_2{E!6?Ar>dDanXSMXWzPQzXy}ND@o}R+Y<)k`iWN2+2y~$mKX12+# zAKkjoJP}94DqiRDq)}5{Kj^->yASz4MD1bGGWi^urk2Oso6@O`>$o_aJ;gjsULD`R zF3yyRD0&EZy19M-ZdG0up%wc+(+7CTaF_2d8Dx}V?!%KkV^_?&k@C8%qsO*j=b>U$ z?&!0CYZumK+(EB>@+Q0XJ7 z>xi`P3Rz*-DP7~TuZP8Xl`;{Mfyo#>v+-WlC9FOrZv89}-fZpI8xa1`H-U}2P@2cQ z8fhAw5iVIXrDhqlWVLmgR21eH`p@+hnS3;P8d8OVuA}s9c{@z6DuY&k1zQ))$oEjfM zh^@9M@$vBik)VF?y%{CE69JD~CEuH}@6#SuA~uIzwzpSe(^>t>d3)N2@)zwa5P)lh zV9%=RN9Je{pn7aO-o#+JDnbc7ewfS&u6*TXxlLB+*4N#~#@ydp|`v<}?MF&MEjKcv4-zJG|ELmjX{+_iY<0CX^X=0k^`0(_yuBJArww;v$NY`4X0g|R_9B0fwe^Y@(087 z{`3eG&8b!k@6{@evZS-@GBSfG4is_~=I2*crEPfKjDb>^*C1U*2(;P(CG}0m({q9} z%J}TJKbP%I4*e$Q<$aRRJ!~xvZEz`SZRKHSPs*>*Y9hAj9)Ev(QKrI}1Bo6XdwWJ8 zz4<~pK7L%XIr#12q=fwTdCLWeapyI`W`ui$M%~@r@7LTk55MvLlSWMspj0fd+Gyvc zq6$x>)2z4Z{rLr?K8+JJQ*S>PF!MSc#yoB=#nK`1rj}mtYo}jc+ObVC^UkA@NrRZa zrJ~{@ozwCEQT5gVRczh=FkX}nkyb)Qy1S&MySuxkrIGHCZt3oBC8fI?q`Mn_`{=#T z_kHId0cYk+tiAVIpI9fK+tmRQd{DgVqeH4U2W|e;o`)Xv+SCAroLa}1Mr@q-c&lG{ zw%6X1T{=?dXsatgJ+7dBo;X#bM*V?QVOa;&zN9!j}UrMEq_0~@lI z1qG%Nb4vY)gEQsqWP^ev5N(SFf6{f1~tYs^Z zZy-2p;^+KkkSd5|FTyA_!yHVuCz~6g(^OJ$3-;j@G|K7R^LcN-WU5C{+7lTS_d{|N zWnw!GVOUJ8>8q}Exvt1^Yf|$4@mt!(diX?b3+q{{BTvZ*-U3CjfCJPTZC2J>3m`G6 z;->HEkEgA&q5c3COawv8RR=+QzJu4;Ak05Fyc(b~&LNuC6Z` z5<=9yt``6avyp3yO-_V>asqh#fDvrnyT8ctHKS^h%UT?H^32MI59FzB=uK_P>pgmL zlrc2;jf?EtC$QZEhukZ(w$pbJ={9#Z+9$Q?r?W8^$(bYji@jIm%V=fSu#&b{&6+zC z1G=#0Scs(Yc47?@2ibs*tL*I(bPh}jyozWG8eVnsv!;g<4|nEjGilZVzL zxwYec+vAM6@WME8 z23%A^d~@I8eHl+r2^mT0lL;wSUU%3uzTJ0oucqzF#*4qCM9OJ->qFTmm4yCYgTC-( zIoGL7ERI;heUdlGePwZ2yZgZUV|F^%0~IN!3UTcgM+LERL`F z^w8Io0``r3WTo{LQqpPao-^MLmUSb=Gpf(!u{t68n8*d}icQ%-z zBYaHQk&}+r!e*K7lx1F>%t%Zauo+cbjmcrGX>`$g=oTW@boDB=^PYBAT2P?KSr8>I zW0(`l%Gb=aO;Ub+V-+joNkBjV38Ovlq;KC&!K}i8XNEf$?07KO0)o{-Br-HKly&LL zI;iq3WcNNq_Yye&j!Fff$xOxiq_Vc5v|w&__6HEIae&J}Z^`(B^{)(Ds*NlH2>H#M z6yMzIgSLY&>u7fbQy0RXmgF%M@t2w$7srRrM_EoEnTCw7V_bzF&RTo2X6%Opl@DKY zWGJ`wG+rETO*J%)i7xWHrzf6@=?lj8nlK~}U|sq+C5A@#8`>taojhb3rp)TmCsD-; zp%*Q8LffLc(9aE9C3vR{!ow5DX7K5y-wSQCeJ~R@Tiaj7yK>X83tiDTuvq*3YtL3au3BgE6yj&|>`U$9XkMyePv%EIL%?^A& zW;0ti(Hr6#9;^$ub2}EH=SXnk^@U+-UDK+k-Xz9`Ds4S_3}&BZ&c?BP2idk6okx?X z$xk?Fbh}rMeK`VVXlCLPRTlOl|j7uBxM{ zDmAjxH}f>(o>q9)cXU{&k@o62n}l$*6vS(2P}`(XNARe7{AjVHocmRD6_(xUlfKtC zbIV`1$2e7EueA#cP2ku!U7{E7soKa_r0XcW9=wz_B8}xQh?GY(Bus9FU?zKZ>J5PM zb2wa39e|_JZq-m<){X9+;Llin1`Px7Sf8BfV%6wx-`dA6wcJj+KrW|NDqJ@0v?I4n zD!JV19)ZtEQmRUdrO-l7qtz^e8kSHnZdtcTl50f}CUiDuURH5Z&uTdr6B+rVyQ%)! zlEM#_?83+)l`D#ZUUtFabiy?zfUGT*sZ1tpeRkE>{l+ z3)W~wwI6Py!z1PssQks`fpV(2UF7T(0o|~ORKMXXE!x_IUSwkdn$5+J^1$x{U7|{< zpOA4y`_CKeI%*{An|BEw=YGJtFS3^!w+V%>G}CKzxO}S=-A~^uolmJTbpAxwHW)c; zt|asscQm=n%lu?Bf5e$R!`x9xc?3o}ECl}Yvm2d%C~=t(;hVI`xYvYe1K7L#S+5u! zgdMYe{nzV7YF>>wcmw6#iz~8=7#~-IjzMk?tTRzz82% z&bnR8b4%jd-Rs%!%Hm_#T{`Z%HK z?D#f+Efb}Qre9Lm5kO>=uWPA!pE^bcvOtvlET)c>EI6sCyS7*!_LyF z=cle&W0*_kL?VtF+f5E0KJKrs((C+y68pWmIfX{W4f4dJwkIigCnqOTQc^Jy%13|$ z*OaIzDRsl>SvfK!?gQk6fYAskM_f!ysYFds_!kyEWR9H*Yil|Lg@%%XeM|a36A@}9 zM$kfp?6$r`Ntm@P06lJ6M|o5)27Iwb^Oq>hxvY@xZi1bq7S57(1nbf$B4Mc%wx}HC zGA-)Gql(9EXOlCge1Uix%^7#*ouI+PqFEEM9J#E~k9Yf3$t;$cr1$gIZG(MvR*Nnh zeVB27g?wH-&#L^6jx3mME*Ha-mWb6EH*b!;1#N;RTf{=-m4%J{0$|NM?*|z^Qbb*R zFht)=pBKR9)(G6Aw|e-e+jikZp5LO8x9wf4PWw0Fj0+Cijzh^&)kntMfXVy=7P9+| zmsXaxNs7zLnqDJi4K#`5{pZu(X?W8%{7(N@J_y1`WVHN<7;u|+)+_nEMB_WB4+B=qmIn;Iu>2|i}5CX zCkP7q255GWyKRX6M`j9Xbs*-Df zi%eh{a)WC0EA&MpwKb9b-qb;?vewR0YTIl($kwJxHO&`|YXMtB)%*q2z#QEeOResl zse&>xBYm-yMTLc(V`{$)D#2;1&H)IcppL!R+}tdiCB$|5i^%@VWd~&)Q;nKCD1&7) zv$G8V&Qf#nBbMN`wY9Y)vNZ=bc0BklrQ4emYn1ayYVFqMT3VLv2|WMMGOZShMdH)O z?c$vUa=z)Wr7)D$f1o~}$LsY7>xKHL*zBB~miyCwz`3rlF1vMz6+3K?tmedPjhWpf zzZ$oB9agK&{Z|CQt1f%z4!bvor0;Ey>IXLFdao({lBeI}>!fel!C@ zcG{Df<@Kq^^1*7W({%u?$N-yqe?9U=u~8TJq98-dg>UR=Swef&?$`9G=4LbSOBfR@ z@IKFiRMDU2-RY@qe8uFjee2S+3byj+0Fvfm_EgFF_Jm9m@lZ0G>)fFnu%IhYZwhX8`Z5V8VaLMME`M-`HrzCg%HIxy!|FFaoT$VTCa_*9gz!byic;HF0LIfm)rFCg5BF zA%R~i=tpHQCC%faRl)yf1%@)QyPkIeHC}_nP{b7#mX@*|l2{1=#%*QzV$onZ<2w$A zk-NqTzG9~smL~gFv(yDQZTe+hx7|gDteF-`?XyX!Dl@s=6EI_Yd{sNays zmP;D)9`cJ-*B5{e!Tkzooi`-jj#ohZyuaMG-1yJr6Q~vdY0V`Df#(~x(#)5js(U*< zv)&n*Z5=+Kr8k^XyDY=^YTSK&JOK_OFzk zH#tjzOd};0@ccl>6D?w>D%)`&hbU|Uze)^JrGs>z3DckSa-?(mms z8lBzBcU8MU$Ijb)O;sN*u6Ctw<4%yKX65PN+I2ZX*5!y)peS2{!HLjvm zNv)5p)GJk~0j%AE!=WnyjRf23!vwb_!O>!}h78I|>;x%%Igw2TipBi>UFiT`y#kVj z7n9^nOfg|Q5+Av^mcOhcV|9lG1>t`DatsPYh@R8=Y!gUcxZb|i0r`7185`R*z+07pmIjub9itd?yYCr?T`9AG53JP zXLB%FxM&|x5`rwVGvfeP$*~s_>ITGXPXu9`={O0;9+<0q{rpCQs$}2hN~c@l7h4Ad zjcTPZ>AQEjKsZxf)_5078QTXiIv$W7#Vj>CP;z}QYd%HbO4{!f4#lCjoMYm^=Qsg; zKV%UvnG_p==0i1cT~+e19}pE4ouYSYY~FvC(*yXDQA(xa??rsm=>(vgpb*f4j0y~x zO~G>Yf!vNo6IN_IdKE;p!LXZ4=Z*~_hdmjWr3u8})-CG>lZ#YSB?3Z$+@k<(koiM)kwm;&C zhMFWfUtFZ~pC?~b5?ZZ2de`P5o6urJ9v6YRpZe47Uw|}6Ny|FlYxy+ElJ2W_v;K>e zO{t=l-QMa>4EkQ1$Dh#-3x!wfMVufHbD-P(**Hs(7!vHI)CxjR#ov$Yvi;IfVL9eHbP zZF+hXx{mvDDl|mB=;52zRz<$jT~--9(zp$Li1O6O>xE8CN<^5jIs*Qz$6&eWwfa@v zReehWxRT4A$j=epz6b3Bx{DRVi^8lNzLdY(EL=HRozlNY>ZpCrIJ|3SFOEy4O||oo z;^?`7Yb!HGnr9^r?1cgIP$qFqjG~f4B3}4@vE#RaopD(UIz8yXr8=Iipl4RhZOcoh5VcK8#2 z_)zXq04VVTUwCu^k89neyxd&5RS@@Rcng&+4+V*+@bFQGLiGl{ulD9g08QxuxLNy; zF}di@KtwGRjwhPOqTd|~QHrbr#W9%T%~|3CYlI1$eY=QvFic|e(p+`{A!i{AP{Bvw ziKSAj&IkI7YT?4;-_g=zsu6@dX_tbS`6;EnJWn^no3A}GKp_B>COOYj`9FPney|5v zw9?{Y195x+YtI9rDWF;fA{Ig6OC}jX!fE4TCG(<>gp_nX{fjv-Ddl3==Z(JDb6*2f z+i6WFYK{6xuH|jC6b=UpP2|SMbFu}Oy^@oY6OPpI$zsyRf9&xs(r(TJA*zS-*BSgS z5G{4ZJIN2{)!NNpD$c8oQG+iw@L8=E0th{3T)J8QriWfaj7ik~6!NMs9w-Ivs2CCl zL<{U?DHkQ1#Yo5H1sQD_w%VSS0#KptEKPWg6KU1Vk1Wq z;^IN+rz&`?+@Cady#jJ0VC!^Gg0riuC$3|B^i0>Cug!hxW2T9+qR&k^$Is2Pgd_(i zR9<*94fha#OhxA!;IZI)@9E2VAsX^NOHPGtKC}~Z1$N1F40IMU zx}*>(H5qZZrDq<$EUdmHOz}mS)?A*D6S%e6a&r{G2g6;ANISo`fN!I9A6tl&ZM&Jk z3**be?Z>1F36Yi*7Z)4RMr55V75}x*tb2p_&s|tv%W+<0ToMissas#SOV7RQ$OWZJ z1yvf2@|ar7gISH%*d7Er;b0*^qMIa2OOU(`aCbr9z z|M=bq`k*y1AGrIhJyB_FPAWe&%ACAmY~X&jN}yp8bNyUg72{GTSQn#6O&i%$X@P8l zqDH<*BUO=jBX$Fm`IeiLd_#f8$cloHFkLbkQl9R0VQcSCml0AT!NDc)cs_#K1S0Dm zO69ELH@7!aD{;I!T%=Sk1>T6_bu%5E6;BY!5RNV*fPlm45HHq#cOr<|3?3{cA%TH9 z^L1@+RsoJv3vxvuxsym{36l!Iba8bJ2nYbf3+Obj#@t_&j7E6^wZM9Jn7(mpe7rIK zw4;qpBE5b$knaKgjE}ds3VsquFxTzzLDs|qk|ra(`*=rebaZrVY*H>7&-)W$OV;h^ zkr&Qai%y=8H^wf{15oh$N>NkUxLWOFAMci*ZtqwNFHD^6TOqPbdhNyp7AGP+77G>2 z;W~ezblgT1d*)iJMKNjk9RzvrG`e%!>(hSPS!J}TZBvM*#t{c(XTAzw({!q_xMsPZ z?(JOG{!3}Eqv;db1#$|qn=7x)3}sEbVk^zLkZ(Mx!=g`G=VsYS1P0(~NTqGVMOvvg zpYI$2KhLr>w%u>rTi9smqYQrIbW$TyqxdaSe>Fvs2pdNf$pLJ;?BB=1@E%%2L%Xit z%7hz;W>aO79&6)6yLoNpb22p{J2uh&HzP2qEdpAasm?bsmV<@&`2D#7C>Oaz3RkD& zk}jV8)*EZ@n|$7Fu&b|;ytuhq8~#<~`T2Jxb)HzK;KxS_Kaw;wJCyrcj?P)ts~vKB z^S#dtKgF)hmX1#A4+tFn6jBzVmNyDK0@({`@;fU5_nKk(u;=ZHWJaNmRzOL&}d_3vV zuai~jMsG5F(($6dsIb9AWf_Z?42Q{^druDW6?vjp)dCm&tD&Ngb`HO5xcrB-~^ie8fsYp}?E;cyOca|s<*=h;58WLNC67HfpN;NY7# z7Ieo%bYwxE7db7JpOWu8l)rV)8)q1H_gooZb+xsK^N6=`x%*kAe)VK?((ZgB^QZxk zM;r`{PY1OBXV-y(sT|GaEdkNa@7`!^S}m%u%)f9Y6x2u(aJzo~EfG6wQNWNCMp8Ap`nf2aY5pb1B2dUN z8H{5yfmt+H3Ah%_vim4PW2ZWv!$XhN$&d3VX0)rw8d_v&hZaokg7NDKk(ajK9ri{m z3DhcQo!x@Z1a+u@DL@lGrT?AGp}r%NGvX9U7$t_<*(f=FcCd9&DpW*>UE`=r76v9s zwYWJ^^7wuUqpFNCm7$(^NfxIM3MW3?l+;+InPV!7jC6~&zYG{yTx>-}d)t5_Xz$&( zpf0^xP~0Y4w~CqqsaFqyYv{ijJ$Sb8pb>)c9;V_ZpbPv)nq)}Ufj5s88&k6Vp!GVk z8Ys;r5Ne{oAz+SES`lo~uhvI+7`AjnftRlVsU^aOP{DG78$Y?1!HB|)RLq41HMbGNx7I5fRDg?M2r0~EGk;7lGlKeJnR*y+eoS5eQ#7E>ytLe>dFC$Rey~M=40OVfK zO;U49%lg`yx3~98_rI_mlow0(@tU7zPgyXo@O5C-G0z)2A3+x?EJXhmAZH0sdl$=} z>nO*DxNeIh85c}pN6h2VqVkpOH-j;$UxKMmqr=`MNVoym_!}hfVE)$e12j00*i3*N zrowueJ5?%RoAKYfs%AGM^@|?K7}3?~cz+(slI+*&ZxkY$kpG+#({syk6~;#Fc;?&||oYFUck6n+{s>-Q}w^Ctf@E_(5#w})V(Y$%k= zgomOFEtMZ<7~b{_IN4)iv3UcZ-AH9hDuxtAgWNmi9nP?AS0Uy%Mvo+nbU6=6t; zvDfHg3RS5f{Cka1B?5vu1$`zACoM@g>E%Tjz?pOSVZl05QYxk<`6a~AD0qN2SYT6_ z&k(1GLm6gdH+alIo!2dLkS#;;Tam;L;PePU2K;yPQtCFS08b8B*b{%M48b}}kf0NG zMD-hpy^yqx{YIG8VFWP(t4%5TjLMi6GN@Xt4f#HmN|0kWicm@zAuii0Cel;=7Z$vf z#;a`HIF?ND%1L10DI+r}#wserD!;ZQf!IC$zxA>KpJrsguhy^_Xh*}tbz(!f1 zNI_keS?~}3e}ffDjTPG%I9f84$Q8sI_-UfY?OP(5_#pRyRU<|tC8k_c*b>q=C|DqO znq>*nW=!o8s-pbNYL&9r%qkRu_$65XJ?Vw#LeW8+AaW0~Esl04Mkfneiux2wY;ELD ziHa#Uq!2vh$bFBiKabNc_`ZZv1Cd{+LZv7qZ|bZlYicN}YUu1YYGS{znm9+ITr)P< zB|7k*|D1PEMz$;B+AXK}TkaoyHYTuh3$1edQslIEsqSPdH8$qYH5GEy+SxkUIj(aO z75;e!ie~wlwLxlRcSA#$`TiMWpAO;^{^+{E6?-Z5TC5%kn7p*&SCLavb&E|87Newz zok)NjnJh#eU!WL#Ndw{&&(0@IF(Ub|Wd|I1&kpghvnEM&H*lP_b$wO@)V4G5afe3! zH!Oanqy5nbPe>ss*Fb#u*%)u|Li~ytlKLn^m2XDbLjiBuKK$QfD@lnK8}fH>v{6E6 zLFm%1Mo$dQDrV~qRFV2~6U6boR7Q^^zEk?V%83K9LLLexRkPmz8-u)i#w_ZgLX47F zA%Z09K0`Sc+GXtV3cyepEeYki3yQ_`saDS!fLDcy_M&*rcxxi?-!|dPdLoqUB*PFi#YH=t)BY^Hx z1bt#pSVbGm{~c&)fJcgbi(QCe`);U1T25Gi6u`T39GpQXge>p=Sn=P*kADDf0_!n6 ze%(fEpQ?)mY<@;PQWSV7N~lVy#hTbg+QTz;EsIrGSYjDAy#f^ zy13K}a#O`c{uGeW?oZ+I1v#srn@jG$yZ$cuS&5uo$_QAxF}uf33%^yGqZ^hXaG{hS z&iaBw|F>xfn>rgh=Tv0&88E5uJ|#PXH}=2(T;X`l8c6{n@PFILdk{(g`^^gr(!k+= zTQyu_*I0fAXOf9_Xn*;DZ*))wSEn~mQ>coA&iO*>h8;}M3#gKLfE1ggB_{S(hOHkR zNuNOy77HJw6RT;Elh8w7u()1Y&`;iWc6SGp|9ff^a|Bwg8HfAA1k=E7nC~}mw$%xn zrZ8cXel2+TCy`AmGIdkfUO?_KF7X?WK(70rn#*|KlnY$~Jw75Z#4M5i8L=D%Ypc3l zWL+#F!M|sn$QXULO@jOrtyQ6oI5xKv6Oo^3HreRo8g^?1uLew*a!dtM6YO^kugoOE^L*Jb{0QB;e|j*60;g5uUizA}B;y(56a8g(Fg@&%sef7)LNRvEz@*eZf7rxLa=a?v_ z1J%A%O$7Qisj_@>2_bWoZbUtUh}nWnU5EPOY#k@&^VRc2o&_ELH^)WC#z_knKVNx~ zR3%U|;O5Ev0<#ykG4UYggjfyv)zW;=_&5bnXi8{LX{rmU8_tEt}obpdUf$G39!28sx-cIY%;XocV06v><`zZC$ zdL&&=1bN!-)Htry30>R)z@y$Jfw;Z27(H~p@>77tJp}^if}rJQHwslN!3z01Mpnwi zav}>-F^sDG1QR8JJPbSgRiO@(s_z#jMMf5Fs759ta_(9EI2zu2bY{{WXC6z7@a5uI zb-9a^ara7n=!e`L{GRXPjCzD;sN;6Jh}V2M`uxGK_#Yg`-u^O1Cq+$2%S8&(>W^T> zU$PXRAo^JW+oi%gHTK}J z*#TjJwLcD159U#z`|SZ$`f-=Nh-g&n?5Hjg(Kp*JKYw@npU~~EO2dPtGN8i|&n5!6 z=*>gwWn}Jf9q?WW5?2M-Eu7i3U(emHa9%T;By+TtQ!muxw9@^jwpsg3DtK$^F=u## z`pB5JYpsSeW#h!+zve%c)jVd9&-6OXt>}|#&WHk#5}t0}KNcA}fBACWQb$XSLhzC7 z#d~1tAqHjtdMCG4Klz1M_$E{mWV2Dlfsvi<&RQ591vrjlIy2~QNWz;%RtOt{7T-*yMijpoG zj3!OwEi_#@G)I=7xG(o4duzhHbldQ2IZcyUR110N@7W!ZkWyEo^}9i{iSw&vPwbHA zO;`I3S4*w{r*`BZ+Iz>g-Cnsk+LV?c8(lBOSa z)B}5oo~tBub6?du$+5A%ZLx^vIiRQ0~$Ejfi-hYvgtiEzMf^;fk;-vmkRPtK*m zQlIHQRln2UIo}S>uRdBM^1k8A@i8W`x}o}@ujH+D&pEAG$8O$#8mOb&yz`&DW!9hJ zg3#bIZ8JLK*X0lvHa#0LHxpFd-r3IlprB{z2MbKJ62c+g?fNVp?DLG=F7Aw>BS{R@ zs#RibdS-C*JHbci;|-n1sJ_RI@AP|%M%krFEmR3HUw?uvMof)PxE%JFe(c+tbP@Xh zh;Ddv-7HI_v2Bw5**PzYIu&!vrb$Bw18)^(^!e1*3cXiY^>PuFrLkYfMxrr`%pmhp z@>xp820cxMsF5Yf#~~|}DO88Ot)8>OKoPjOe)Dj9cA`sytQ93+M?#a*Bmvfrgs-S|>b+EM;wyi^CFj2MXi4^POldx39bVw>!cvVj?z{uB+&z zV{VIE=~czd0rMHBUr|5un%4eoZoV%rPS4o~{xkyy)jTQY?(up68Uqk~Y0>pE(0O$< z8{N%TB5?568;G8|C*7@2Dst1lQh~q737D#D2X;ps!8)T7=P1^6xjPv`{l`)dvM@0r z&o#xweOIpO64B9gU(-feJ|;h)xIAc~1D%OK6SCHRFz%l-urm5oj@^Gcff_buljwSI zxA}UEb+CUz3We}PY5zD93jF3;l@weeLF4hNlILyX1tLzrO^-DE8{4bnt$It&;($dR zTRC*O`d@*h%zA0{$gi1``uj}6mwqx!huMvX-{tYtQE(-X1=M~rz8Ov+FsjiXBKo3H zpMWvpAcTVBXdOH>CKO6&aB*im_tUM3((VHF|0cmov=!U6A1c`#dMU^K$D|1(kq!+fs}~ zd}``jSGfG_%*;&VVFD@-Ne95r7^HK%1L3NGDu+Mdb@Z$7|ITJ6c72p#$YI=o@BE(S z$rSSjX+v-}NWMQ_A}p|Ns;S|-4rdSbn8Pa}`qH$2^x0`P37_Z5#JQ_51R^mO&3Y{UNsv&85>&zr{v-smyue_^L{S<>cx#I9yV*= zqFJ(8jpMT(%Yh0`Zy3Z@Wqn4h+&tI^{IpcNILd3HzCxVX% zl4vl$;z*XaqSzU$TQL3GTHZFQQYzl^mjIsANC# zpOXX15r0%dDR>ry05JU&d)EN4aP%I~w(j@V(n$DNB6ea`D>aRcj1DeBLN+!hs~w4H zX#~LA6;FeSg^5`)I5kz&NDeX)PXPM?=n;&=4d5dk2HVhP+YhsR;2Z?t+`xdu9S)OW zK_r&!i$an(mbeE=xTmA!n57nG`;*}mcFyaJfKC^@=_+Gs`pMJnUC|bQ@H!JKA?-2x zUvW@yuyC|1YdiZPe|X5C{naT@u`vx!_F&%$}-6sDjHbXW*rMEW%B59|R>fxAZ zk5aqGvAF5ORvO5L7>7sX55X#CT%@8cAPNEj6AKHXSOOH=jmtR6o!085tf`nfRa|AI zZNc!xM!#V76j`d(^5a8Hj=96UCD(y#{J&y>dAX-3)?#Mi1>q8St?Ccml$PB^HdD|R zNJ+6jKnT!>)gTTcv}4asN=&wK`=b8*Z4r8Wu*zeoYLu2NbR{>jFG#8HJuCLHM*EYV zl81|&FA1wvXnNj}xYE_KD_Jlh&YUB{2^#kFcSrO`x;PFrZTU#MgHruTjM@`}uf{!& zR2hyHLDJI-SGbe!-6{@(Mlf^!^>YpB;#c`A22p8rY~#KS)zi>Ftql>YwmTZKVK)o29ONz0M!nT% zCNeeR?_uXeeZnDRWMoJo+??sDMU5R>q-C?U5px{s{uOTCk@@Y~W893@(<Venzw_p7AdmL>%eB3^EexzNjC9GRjQDWH2+c_NiXLhGYDj4yT zzwKksWu{AttYkZj-2_+3u(#wR!C3m48R~&=g0GKiKs%T)pKU8kOxQfZ$_Tz`G- zOV3T0t3!9tg5viUw#B0rr*qxc_k_KElel-Hj*15(9K>cF~a3%Tn-Qfu{<~JWc?bc9FIT{?Q zTdS2x$uIgSRh`h(T0K1e?tN{|SYz1(gxUb2wsUs2va+(*iCL`k^b`^jy1R6XD>2tU z7C^1o(^>9PZ||P2UBc&-cd<8Mz~y@1kS9HUk!=Cyp|8-es01JFi8u@Y9hXI87Hw|I zsDYtrJ5i@D_I;{-H|7El4T2}+4Hmko3((c*K#v8Fo1@2v{KB~hfo(8#`k#`8zk-IU z-Ku?hm(cKaTZB`x18khlo~|y>gSQ4N3F(dtyfnIe8fsxLK{`4*U-81H`^9a@3vch( zzl;L2nLx%tn5Y?mUUl)$u2!TE>R zqkR_w0)iDNCmb!RoeJ0KBLD5OX&H)9SCKfax)dr+A zTnuU|+1hWI&CGH)=e#g+D?HvYc4}^;8?~hV*m*C6_*1Vfkk}43=OZDTnVDPgXDCy0 z_|b8Gv!mDhJU2X}bN30d9*yK@YteE$UhA3Ncjj9g+gQ|z`=m;G)g5R0Fagw}tE~Nm zV?=1TUq>s35$_JrPH&s}R1yDipsjcy>5-lstjGCm`FvzA1?^G1T-)37L@3k+}A|i$Zg$Z*yi^g+N@v*K4s?bl*5W(_(xIw(s z5);ZFH$04B5aej(O=xoL_@wxjT3TriopB;f`X{QDhiAj4U9N`>0fy!s zO8X$wbf~R8us=F<221YK-EvqlG{YOrThMcFiimCgJ2Hyf%GW2*(kQjQ!>kzla(?uv zTx-?-R-ETv1VItiWBVNo6r5#%rP=UO%O&f_zkPDOYJX9r$=hieJ6A2;6$y)+&o_E2 zMlUa0tt`L{K9VrnRBp~xmdo-r%Kq0?LT(iwvIOrG|~g-A9^-?EPv555vtva}|C zQeH{N!;an0HKme7Lf$J890!R8Bgr6`pD_7!yBc|NXwC>K$VJgSe$iulQg_TGU?Sc4 zJccniT2H6;n#rcy*1bad)8_-vxguC0%Zv^fk@x zBLz#T7WLmyd0%ppIF=dNuB{$r!ZcL)nYF1F1Vm@LSKU1L4@)uJ%aiiqbUZF%#BW#~c(WKp0E3?P_xL(>i89u#p zWdd4eT`K30Xz$p7wb?dzhsgsV;NMT+Y?c5`NNty5CArdW{0{m| z6HS`&rHh?N6T<+_22zwRi5k!{D!|_#;tK^o0ldY}&rg8?$oqj5Uv90WZYlHp_F@l` zW&_W0Q$1VrWFd6%leK&1Fbe8X_k8nB6VK4S?t}Z^BnJvgLwU6ekq5tx!szus&$j&m z1NrzWi5@D|ablMs#Zf`5wXUkSGtsP8(C5fe5UMc)Pv{G1^YR3d)smC<&d;k;6|#Vq zW&phM1Y%v#fyyikdgMYm9m-x-b~KAfC^&KRcqU)=Y_o;P=s61X$at>ga+OnA!q+(9 zckzlnq3lHZdnwh-e%Q48+_EJ9{|jkZ|3w@=hs!-AQ4&_nL0ztrl7gY2V2UC1qLPAe zZQUQrUZCb1O?M3tQvN-US8Wl5#V&@mKd>PP*cjx1VS@6)$7N=3f= zu(0P@Ba4%RQk-(awe^Tu9O^|k&PW*jUlOEx^Q7QkI_A3-cpmrm%`sh;IC~;e_>k-I zf*!>2lHzm;GQ_u4BBv;lxEve-jgcWV)8B1G3`48A{Lc3?dOJ)z5NV854q%B}%HxaA zHvV@_BVse3tasV}Qd2bRqbZUsrn+QrVK>(L{ zJb)G6DwLl#iG~NghAA2}jEQ(m0s_sNZ@8ZK@H^kTZ{%xEw|Neaf`?~a!O za;e!H@hNY4z9gh+c(KLB@o4vV_ABiFp6dODf|62dZ&QN%+PPnQX4?+I{^4jsq&Q^! z8EPBaym5s@6XYZGp*Tt|2}Z>H79{YP`F)Z{pl{D-KVlm@1YDK^O6K2S^&NyT3xjAJG{ES6=opSynex{d>NT{Dp4~4;GZRDIUSQn%*yMorhiXQ40-f3J z@7;WUzTLDYs8KuE*>Uc{cM5a{=l06&ZIjuaJX4|{e_0G$4c;7}2!DLtJa{Hqf^_$; zy9V&OeTMhQaD&YKMO%`%(69eU987|N|IEe3@oz0etD+P9><=}Uc_Ff z81extMo}m44DZ+_!-+mxghS|4y2JsaZ@oL7iFD^xK7xP>=23_6xGOOI!^6`iOjMGd zwDeSzXyQe)7?s7Js7j3Fe^tJ!s7_K43HSutT?uxDVnrkzt4KHCClh{$&CgMrL6G!v z9l_3TJ6KJLp~jf-A=_97*^yk_7u*CC8& zJVsl_@84OjHnAgqC3qA*aI**ybm|Ne027&ssgkxcd|@*Fylrcko@cnKiaun7eP%4@ zN$(8xiWTXUk2Plw68OrVlxKvG=@Rt6kz8cgYWKp?ZZ^e0;u&IK3GI?)do&Rl?z_Z zyjKd)-GE#YYZOMEF8p1!1P)2k(WQmrpSM6&MR$n$FlbOa(&>36NPO?4w=YeRgjgq* z<5|+*6$_dEtk_l=#fpdsAX=3lgG61H#rkYhrXi8Js5Dc5=jCr9%dSE0DezB+qf*gPv)}wU)k-NhAh;fwFo}ZY4jT0yk+k0r|23HLRztS9{Qd zkAa2sWhHo?df+2}NRqfwpsB`}-zm}+rvy5Gbr5A#!a9Q-&5CxXVw=kTpq@nnc@(EA z0Y(|=c7yDc*pGFZ2w;cLc5t3#LB--^vI+`JreoQ_QL_yUD29H&fqOvV6+XTt9ZoT& zxLtrrRCNq-b}?554LIt@pwXKa8@9TVpnoZKFKD^n@q%CB`@ezi*lTNg^hA+yl}Wya z_}~8iQ|Jnz$9X9O{@Y)_Mlb-PTLr#^z(lheMaTS%*e(pnKI)JjypZh&GZm9W@p$O0 zMJ~Uzv^3}>13C?+aySIXBJKsn_V@Q^Wo5x4l1L`m-<%q%?E&nIkkgSWc3iHo3fMA$ z0jBwEh?AAqpj3ITsuH0+szcY=+35z@!{XlnF$C@Glw@V20s?vfUhfL}OoApRY8yYkLRX)I z94Ch4#fukdX=#Utht)5-Nnj*rU?2f8m$He#`Gv_xq?#mIt@O%t1x&Yl1n6 zYQIAgHyXKd*M9nv_JP}*-;LOLtI}$a3%eeFMb!FF#jjFSTaZseWWwO~L5J5+YQ3}l zeG3Y=^HE{oUA+amb|FE*2LO3AIUOk#P1Y^gihFoG0)oNR15_!65+NY76PyJo6~HfA z0BwDrTXfP?MuG*owrajGWZt6?8`&cLs~~ps{i$KrpPeCdc?+V9;!^p&8AEBn$xvt4I&2dVn!N)9_nXPq;I;@s0|=l1&4VWs4*>%K9$pU!wTbX4deZ zOiFBHTA3lfu@zvE6AOq?w=N%#B z5a{Lv)}Nh!0@keq$%`T(BZODoH~HmX`bVSVpVo`j{r&s8wg*YIkjimF8K4IOjNCKyUvXnx6{{3b(qvKn#^|ew=YUNx0 z`2fP3X;wCdRh=k07h4Jc`g%1;Pmq7yC(W>FpXlvgRf+rY>*DDSb{d3-OjWM>U?1;c ze(-}=C+m|6e$DtJWcjo_@Y+YPouhddVY1<*Jy^kAN`!}YEGJe!S=~sq$B>AT2q`w1I9@3S={fq@g)m_GofSn$Q$Psj2ARKxM==A!p=IV%B~IfC;}pqQj#Lw z(j_I*-3 zz*w}k;^nHPaZ9tWJ)+k9nw?6PE6<`&I{y%o)D3j}s8|ovX?4AVo4GOmli#Ux8qFMp zAM=+svO@gD&f2@vR)x8f)Nip_SxRH9jfBoqc(WKN(wd)z~zBQO#-;)k2jM|pfW+X!2uXK2S7cn7wK}ov%b44?gI^n zlu=VxSLf&XN)^WR!td!f4PioDk+(*IrpWJFC$NI!;_FEjo34zd@4c6%ovoKyI+gu+ zF$aU)4F&Ufj`jPO1($1Kj9pdav4z>Kk$62z*k?{uEGuC<*RyF~##;{=o}{s;%dOaH zv;;LT!^?$^E&M;o8uggOtb2j*1SnBfb_%wy_DSjb-d_&V6_9L_ajQ>SDNPL z_)>L3#^(Kv519s1(9g}P5=Ij~StPD_XFTaN5qhHj%1=n#-SWo$YBq-=;bl5%>DYz&+t_kk_jC(|)2Cr?~C|{cd`wQoX ztExWNspgb<+mLj)ke~er=$NgOq5%Z$+lD|+mGsrl$A3SZRf0?U-_?TL$1U@PxWm?v z4nP$H?9t)W@l19Xa0lK$5s&MSeSy1p+}hqgi#hUxJGel9nF2AgQxahTy?yTEiFRkP z_q<;&_5MBRWRP8(2|eT+lsTGi(~8z8o`9b>3TtAQDjTpUIV=)3VM3W~@au-t+W7-7 zr`1tGn|CY$OyKX)pKaFO(Dn1h!Igd4u~)QnT=AH`yEz{yT#t7g@w0y|YPgWtUSnFg zf9rd({DG{ElxEU8_n4A~%(P48#Kd-B6kdHo*avo4%Qn>pt*U;m&Y*fYb;o^rh2(-? z3DlDl^8=Exl66p6dB~;8t-8-llRwr(7X6CN{Quy*xBX75>K5%_{uTBrVNrk9hF;vh8ad`yFM$7pGXXUvHeq^B=n z>@*NVdet}?T)oJEvbl{z=WczSUZmyhSw#jo)PGt+=f+TLj_(%vbJ{m zaZEE2iE{VKG50AL?F^6<3JIeX6^IP> zVWi>l9Sds>pcDt|`|OWTLooj09*n*5(%}=)=*V_u=IPZFKDoAiJ5Mo2FtizFz%46R z>i=mUqRQB>tt2iR`2?;!_a@cFZiv z75A{emXj+cb?-pKKn*FC#?T`?sFl;#>qJuiXQ8r!y4lNZg-Mtw=A@!74SWp?I0H<@ zfHOeoR6!6v`OHHy7)8fnkc;Esit<(m>$7`9;cBmjvUHEZW3w_ndt^6Z7IURA3)b-> zr^00~CV8@HQ=1JgI^^BS5f^`Rc{Prg<;x$3<5MpjxVf)dm!)ey@cONeWR(s3<&z&C zp@(p}?yo&T4=7#Qv91v%P<=oB=UVu5hP3eBWb7s<%o$(({(8*yML_C1Y7QM)jWO|B z*>q^P>*hGqy!(>z@5{%duA@np!>?cdlI)={I#a=J;oxMyCuq1Gu1W5DQ-1c&d6GRU zWd5Uh>)+$R5Ui@Oiz&iG**75?ljR=Wr(Hlwq470C3WwCb?qt|RQW`^koMsv{J>$r8vb`obxW`wFc#sVzYu8u9|ilA-ja? zZsT);#zJPHRT}|Y5eaDr56h#54dEqJpEMNcaR3(TLe$L`C7PXDX0ECk)(6u4)QIDI1^mbL^V0<^EpvRCe^!bg%bRW}?_!@CN8ko5w3}|9e`)S9> z8!OmcrPBIIMR<)}Bt!H`ZQSDy4)7)i04ZWUheZiLzrDLyxIuT#giYEVn%i7?682^m z*>TSkbUbv3(5S^p>3Hgw)0M46WU{-M8e`+Lvs{4M^g5{cHK*-Y>&NQK-?CiL*Df%U z_K4c)yJitX=UTi|QDB-L=|7Of*=MbHGq!d9w|M3ucG0Z8ix!{BR>laLlrMg?<(Uz( zhdD}gB_mNQ=Zfn`o$mJx(C_Y$dQ&;&dQwasej0`8LPTcV<-pnBZ*%H z$Sc414lt(A^zsGV4;JFQzG06vYSNPJwo*Lw+FT9?BQ3u=p`UaX|I@&6t=kjD$;Z}9 zY#PwNjmtPuAad;g@|BU5#9vRs409}BpN4d=;?qIYC_N6{bDeAbv-e|Fwe@(nAS zvy4*EMXCQ4dZSd&iNEaV&*xb`19!XAtD4}Vs~uH!XQbTjgoL?m;8|bbbvLW>ejPAtKJzh0J3%^@hn!t9>sZpNcwookhK_MsHET3I7Yeo!vaRjqe7R0EdI`LKLQxk-|{ ziSC}G$NopUn|y4|hRzR91KG1uq)-V!mF3e8ih5YpY8mm-@Uf|%{>MliG^SKc7B0qD zZqivq^*_0ApSR1=(%nL6;m3pHP+~*(2^)ySW)(Hmw4sURF?}$K;R!L&W7X!+2NVVoCkNGY^V@8C54 zC*9JwZMykRR?>-{s8`@&OimtVZ3nXjiz(vgBjiUam+kKC4t zME(hqgzl)@+DaeCsrp=gdjC$7iHqCx`^2$LrZ9+C>vDe^ruT!aGV;X^ea?ywc;QnyDu&EX9=nWdf#li{T1N|w8G=DXU<}1`NaQY z+>1Z+g7)xufRuyMc9eN2(ZW7Rm;_~SipxHZL5r<|CjmlmL|B+L0ObS!tHaKiC`}o0 zILi`!Zk9r$>-7oHTLH5btI^QA(kCrt3j7H@y7PIxVjuHoP)w`~(Z*mxw3dH96e|u3 z^BGZWKs#@jBC0|RaGt%=27j@oFTskx~`v^eEa&((Y_ul{!h~b9S>AbU)H5Te9x>b zC5fle_Vl^i9z(^hIR;#h$cq?IhS{xIx|NWAG|5z%13VQhO#|ZV`qR7)f@dfgMdeeVDYA-+m>A#=< zkG0}|eHkf&6-7|C)+QOn(H7nOUciauIkD55k%6>7JF3w)DZM)JGdMUCGBP^a?s~5Q zh6vC%F=L%V4He*AwiNJAXiP}}GC+;(_HUKd4wH*h^B#CP{JiOu{AQ(8O_i!YwwwxR z9B$4|PBkhopvI>HBeZUH8wv0li;TnNG~b5#!s9rl?q$km1^9dJu_j@ z&QA+c4AYA;_*r9M{Z#e%+M~(<2j!uEb^%0h1Y%~3mHVLLQBcAIp!KNM7KpFc*R{z= z-GDq6T*WbGGe`ROLZwJv1EfMxL_`TCIh(^(`mckIx&2Q7?$WD}TJ*z-2f!Jno3M zlw%T1wrQ)6+)VapT)K0b2*Lkon*+?FwlHO7Wo_-WUJoe*Lror{q5(0MDTRw3gGefz z1~XsgwHro9z?JA8YXd*?^2^u zv+hLZEfJhJH)qv7t&5tG3XOb|C%Rtnqe5xkOez$=e9V1Bh(cZcmzf{A(XWL`g10<( z@e%0To`t0d@6Cm0Shz0r3EkSHViSuMrn72_Hp4(kbC0EqY**RszEnB5mDNwgoH*W0 zOO=FyA&fRn`8^AZuCXMf0je;*05FDrf+31>1}kQnj^rUi(oj3yt90{$4}Lon$lpvZ z8oPTk$=V+h8!EBtTCu7tcEM;2@7a$>(rrkq;G+7fU7_xwmDj52Fn`{hCKxeQV?$0| zYJp*_>(}-wvq44sl`&;J-a!OaNqSmSCpU1`n1`4AdT*9SZ0IvN4u~7-A3qLYp^FgW z?-vu{Pa)9Pm$=Qc-axUYczD>a?d6HFWOA2{t18u=AjO}-=I-xhkRzKX#vQxmpQnU_k}hV0PoPtXdfC^2b5Y4;)?iJ#Psxr` zbp>Y&gFU&=H5F@r)2K@ulsjD}ai4DX*8%QN6Zi3p$f3VO0XqRRt^8=0@J=Amh+d9KE5X;4?l5o2gKvvQK9;nkEMryt?IiK$H#oo3IBP{&QAAcZ(# zvPzs*08-~p?mmpss9FdDLih-jj0X;VF&4RGTD0`4yU1sHor^|M^%O}_$yfI*U+=Y4 zX9MqR2@SS10}C ztUM1}cKo&{UhA%9f(4mzP-*+^Cg)xIA6PJz!!ZCD7dp3tyojBW3{$r;JOye1S(bJ2 zv0STm`tDkzMvKI2P9suY^pTvQPJv<*gCtnFj;3qrXIc1$_sh=JbT>*K@s9~XDr)-w zsRmjO>AUJaQ-2QS!F}c~a43NM@rVzIE1;pFa|<6q1-@PX(-4vC%IEEQe;j9LVrZ6}cK+u6GD83UWu@CAx90nc zHy`3C_@gsZn~p$r5nwsA*MiNxvTZLR||)TJc$qA#|M{ii|w%UD}66nTq>Veb!aG+?S!9r14*9b<~Ie@!<1Io zi&{xGF6~OUp0YPzH{GfH9%&Ybh@TnRWWf57IX4XD3uz4TU6>=0PW{@P`g-5(Qz(UZdTYLFWIuCWiWFJ8#RxB8ZSozL7JG|^Gi znn$E8><#(pwv$4GPlaF7Uo&}(5`@R_>E>(u%KU0eixE9UQvirR0W>lqB$+l1!Es*t zGwT#Y-;b7M`&^-OdVG_KX_h4(4=HUVkC_rGvDC?W;zgp;R@yS#$3A)%p7tBb57bhd zY~20ad-sqV+T4KA`_ppiQhuhNXQOu&y(ffshIdg4_gOwV_zs3#udX-_>!6rx+|9!3Vl8ZDw5F5n-)?#yue4na348YD zjdzvp;g^|#5v!2V0j)txtKL38E|4MrM;j&3=p)EMJ5yTQ+!Q`!8@a6VhQXhzBXeO( zbMOKAFO7t61dQ|e4xd?zM@_lzYb47E$}d!@k#qWVZI3=J%M0@DX(oFDP^43AYkRG) zZmD-092WHnv8LE+S(7_<(kIQUxWO$2K~mByA}C3al?NYL{t9X+Z3~hNa5Zik?$?pZ zDtcF2{-|eS*%QBcL`MVXIXimdRCOAhQ-$Pi95n}pye>q7U+>%UJw7M|!#EYjt8%iM z1X47sktnJ7)k=qsr)J*YTxNd;v*;;AM@fZnf?P!!3^W&z!9{|vV`f%xuPtxKJinJ% z8hB)?wEe=x$*D9m0~IWMN+lQkCRRHKuAPlZVPx@LQuF2m(JhB{g|}M_V+wL!qVSk= zsos({%AE+I;}JY=rDWnN@K89u3vjI+!Rc6&Wx%+qrbHGsfsD{nCeR6sUz=gfM0TE$}{y ztQdQ@nVv#KTtgA%PN?tf3)SQk%$P9^7CwJWMx4bC)J~SX8V}9_8VMsu*N8v=8DBW| zV?s0Z+qvp_v$7Ixg1}0WrP@SEzu3|*9TNLaN#u!2>|axdY~Goyb#VY{sjv{}k3}xH zpP{bg)2iOu-||n81~)8NuExR*v*8cZ!8A__S_N#Aoh$7LK( zota5c(K)LGn2s^BlyFAAd<+znBi|!|_r@u&1JL`wmP#pY$&`Cs7zH3RU8sme><8lT znZRx~oDP7HfQ-s89gmT$?Cit&x@aD=Aq>4mcNRSXHMm`OO`6}mJ-fyH$^I67kyXOV ziky;+WzO$pqRbO<9fFV?1hSk0q2FbrOscHSD&ib~}_kU9$wy5m(5oBtaBGvunx zthQ8iyfjjkxryyVqqEMsqgoiZI0Zad43e&Q5WMIFz5CzDNobr$YF|{D_;x$>wHm1m zo}F9T{nQ*Qk1^PV0UC-jI-Gt9DqJ!I~dJpLaN~!%c&A`>=S!rpnB6hS};0 zrIu^^?OPfL0^4jfIIpEm6x=xCbO%$YJ zY+FXs=O6aV38(AK(?(x(Iq#JbekjZ3X-ceG@P|B2Nn;8>@~)CFO!!;(F|AdpBLLUb zIfWayi$B;kIm?7KqTcG3kT!PKmG2#`t`;)S5zX+mcC`ETBEFgJB}?Sk7r7tj6dB%< zxQhSIJFL1AG$`f{kkVGLTKM~PoY+Ekl~3{owHbvU{e@6d>&VZ&QJ7hD^YFOmVk*A^ zWTh+UcFXZtE%4khb(@a$n?lOqt4C#rJK;?Xr!MDMZ)+_CdO}qD6sqh-DEbN@8L#%D zRUoO*Uz(=@ttBkc1%EPB>iYu$*2F5y{TtbPa5uBZId&f#i}ZWoNBddfZ52c+_(8s& zf|fMNh0;5?srf26WC*iu%OMces6Tu-uoLq$@5(kV-rM9jO`7(w@tLFbu*TU|x5Fc3 z?a&zO%BWKLUDbUY0e+=#Eos#`(YEPAa5pDxsN+kW=yG)6fX-&nAv5nq>$U6v{vrsc^IYjxn)N7ESKPsn0tSJ$C&3YXfR# za5)(Q6i#r%91s9CcxPgLd84fl-tkXepB+fQ!PlcU!}w+r_tj_b({rGprT zZkyHN=TG6QMJieBc`yQ2Z*F2_lw@>Ds}erg6I`@Azg3f zf!|*|q?`OBb_kJ7ggVVQlIrvuLuT34UHmR{eBL6Z_nsGYQ1|;Io=kj1P5doCHqYyN zQ{aH6B#!tY;{wyQ}j+?ra zP;)3}J@-wPO%Qhu!jHE8YOrGRb>xU~((BYKtJ?zQ;E(~j)g*NK(#eh)hi2vze4d7i zib!zK`Qdb~X5JusFDtH&j|4vX6gn%p2yO&Y{6%{$4aZpZn+l8kZZNHHjOM`k(fw>L z%6+@fwZup^oi}qT9ahr0bZGZ}Xt|YHx~dito(1XbSX4}2z~qsktP(3l(o{4!AOhC2 zqx6~xwV?;FcSVxnR4OTp%VxCpg5CWxO<1XC7Rz#yjJQT>CzL&V;N3jKYMIKbsYtg* zVKniSR0e$`V6;?kY%S>aL+r);`o&(3@F9asCwo%S&-@De`AM4tH+hzzWC0{DS!U*w zq4xY9C*>{mHk(^Zne(z~*2(@mxgq80f)F=@O6@=?n}bsq> zH?BQbhe!8&j-!Ld)JKbU`tIeH;Ah2@L&DssWorXZ;Efjcq2FnHim3Nl`@s(k4i|hO zvskN)G)&81b&Zg%ts!ya)ju+hwU*uNqXtnsL;fLmpoNh8ig~o`1=u^0klhG-wcga1 z2k1C>+dK)ReTehyCZRk80mU@c$zTH_JCEDExR|`b72=;9kNef%p^i?oE+7`RhudbL zTwN1denXpLZ*PIWPbF3Qnpilf#68{+)PNp+q!Szi{ovf}v#X?|GB1_YJ|l8dXE3X4 zQf06KpSoN0LXnEdj|X?hWgo1m$?OSPv`QA|Kz%O#@f?kry?(pgt)H!?<2SJqWa+~q zHa?$H6{w8q=%75E(=k0a`e$E?NK%**s7Cd~pZL6is<5LSp5cr4^dnf3&ILB*^PC(2 z_>{P@!N$o8NRCzARTKU~^7tuLS-P`nxk&wQv6jt%eYNA|C)owWrmU1^Dfkrm|cb3541u~hc$MAe)-lO z5@Y7sz+vABN>=z|1=^I9rH`LtW-fsubm!sKL)$>!$x2f}qDdVr0=8Z}BUKCn_9gJC zJskIDb9qVX}Ta2EIkW2CoAtS+@Bm6}*dv8$vHq5VvQQv8=*sn~_!q zU6nVpTWSj0Z`=|Ltg1JUueO2hga3`TwswYk86_^!M(B3L|3(;tYOj+s~v*d5)t zTXb24E_(g(-Y_$e#oIVqJXMB=K2J2gOznPfApiR}%wa#y8zaNhd~Kry-f8fpA)q?# zVd2rwkkYI*iw}(h6^A5X^ZZZn_lXU91Y~X~KJly3rwaCeKUVnvTad0GjWXGAPBB?A zVT1nPTC_P7p>?GSuy@2hCMOTi#g%AK6dRyFu2U~Wh%o=}4;JDq&c>cQZ0Pj>%t~%v zT3K3IURha>j4GuhhN4hhf%pqW#ZP$uD1!g<3?-yO$^9F7d3ilOKVNyJ_Q?>fvSUGP zIxqIQ?Do2{)N5_Cx*G)GjCZC|1#T}a_BAxeOdw*JX%W@!xPtcbEelXXJ(@Kwi;OHx zOyT;to&T%?8oJZ4Af&EHe9}TU>-)FCg#1;5;-|S4^N*Yh+rw^Vbx4(ZA zIE(hbcM)ooJX?K_i%M{ELjT@vq?}XH)AKr=tbx2R8ylOaUNI0j zR#MUeR}+9S3HZanXUHcdg%s%zijGh-PYUyYo=q(g0F8j#9O(D}Hu(y0-?^O6a=;}F zD5*b@<^FzB^ke%sT3T9wNS0EkR0YJADJUp>7NzndC;(C;giJU@ zoU=dB$VSXKU_H}u0l&Z;_UX$GeF30%R8&+(u7`&Q!1%_ZW_e}qG*O@`IVZxs4@_pN zA#a!H{k2%mW_0LXO~)9A@>L4PD9&0Ecs&|pYDzPr_6>md|E?xhA`1X%BO)VJX`)6( zMu1~=hU6tFg*td9Ik^*%r|ki%PXGfAoNAA?4u?DzbJf8A)YjW8o+%gbB!$l*boMBX zku(S&)=)Gq%KLz8dA9%J3A^Pf6!+1I@s?QZcWmtT{yr_8MFgNRo6pxO_EnS+0>T1N z&qh3PVpZXxn7hDr0A1r%etTzUU3Im8AZ}58y(_@r>TdXQxq&U7vopx-?g+*`Za{}( z=E{_NQnLbldC&UElPk&_Ty_FsBe2hsf@c>;d`hn0KJ5muN{m|_M{4Uy{C%-gLmFfW- z7l;_5K#Ac$6eHkAxxrl{BG64DbDcl*K-Rg^#)e+FePL}Ek*Wh{7LqqYkYV6*c zt)ve|9G&dhHytMK#~sDDcQm_T1A5=)WUt$NgeQn+8*d^!lXSSd0;C5&;INKa*y{pe z(wI0nRDxj}gUKEjd#bFiaQ!Se{kWF(Pt;jJJ_TunU}}O6Z#th)FZhPPHOMqWHjA_$ z*j{XmU{<)7ZLhbk508W4JMMyExe@a&dG=ij~d;{{U)xJ}*g4H_zZ8>%oG<0nJ8C$X5ZY zgj$)KD-ZXe#fMpk*v*-yqs6i0)3$cW4{dd`dJk0=+`s9AxB25EVYgD|Iq-JVGYd6p zM#je#_e@mE<4>ct-2vc<1Gr|Q+`t(Jgc^f{3k@wTDLYn+xvRbM!LskV#!^qvh0$#3 z@*c$Z=6>f|2_W1Y@YpEDeL!MkO20eMWq)eNnGz{B(_{pIA0a|e^PC0=lEjSj0;$m0KYv0iw3x+NdKgC0@$`7W(^Ch!{E{fEK_6N7u%zo z6Y7@cAe1y8EFdz{W=n{rhvm6FU));m!c^ny5;9$;re=JgUnm*SNkU&NBx`pT=8hS0y2b?jkPryw>$wBW8<15 z1m{=@)~{VCN`8rrWhYVOP;B{Qpg+& zB3q3?1jS5m{1T~rT7obNM9oJ7x&Qur-6a^zAnifn6JaQ@npy!s@iGvqeEnuNi%2w?^}(Hb!HPnxM74 zQpO1uBF_oP!H^^^cI1+hH`Gv21`#_*Oa>wARc6y=J_uNiAiD*;LqQtuHSiS3$;kmj zM&__lh1x!Fj^vU-LP4McI4YvDfqjG(x9*v0^?J&O6W@C2!&zth-Mon2YkdQzMOw8j zA~)hE2O;Fd633d_JQM(C28*#*Cwmhy@dv4oYam`t%H&Gnjt-XeSr!p}bG_4>MK6%I zh(KVw6mr#f2iw$742zsow!tLM4UJ$B0l!ERmjfN{+o(u?YUa1NALI7Lb$O$h>+Vw| zB{`EorZW{fXobYRHLt-N}U ziP>5B@>$AT+1VD~B^jznhk3h65Zxo0`~$2|RUVHI6&1`kuHUYXbig`=jm@sx<`3FJ zcBZ?#dwyQCaO0L6%&sv6-V-LUj@#PE*&Xg301liU7nVE=5LmRX4n7?Sn7!RXCr9Tq z180M;G)eYrO8@kw(-1XX-SDp_GqDYlU5(X7F5P8@=w9%7xHd%KT%Rt$Bvz8sHxMj z%Fc)0nn~D472&EY_f6G+{+xCY*KMrLKR?(FekPee^&0!r){!p~{!ew(=X^|QP1n;X zcv4bQXR=>G9Lm@Um|9QqiXfOcwPObaBZINw0Qmo)`!Qa@6s9RqaoO+rKPhtn_y98m znGe@^XL$U+k*hvIQOrhCw;%WA-$msJba6~xl(vGFiW;SXS|5m8-)?8eKX1?Kts!8EM8ygU_gR_-+1e5?edFOd-#n zX@9o$Pku%wZ@ks+@*9PJ5dc)UzBWuLUnMgSmw84A5evX{17621*yNRCdYNYwnEQUGt8Rva`o4kF&it_cbjr#4Z70~(9mJl`kBq(1ZNxabxX9wov`28j zCVmoo*xO%#cabq_zr+3vX)vS~1ayMkf;B_xZ|h=I*=Tu9Aciv-m#r*#4)D7a!p#>`!P{&X zGO49U3u~SVT7AR~)zn6#`u-ihm!vws%4giF_b>i#nwraiTP)+^C(qPVSP6idP(=cW z74?S?L#?f#HUDkQXv#=2a&kyuJ~ippQ8|9;4kAh2F@q(P(S2NLa(f7Iben>}zrfp3 zz8b>;C;NO;mE}7GA1!W63?>%+L^_<~hY&V_8wrG3b(%6JyYufAdYcaQ$Ns{?Ecwzt zi%kzD+Vq#1x#VD}?!Z*dp8kneTTe+z32x6l-TdnEF*=5jFJ<0ukNeGVoF*p1&-O0D zEs&jCX9WuvaaHImy{kh`JtxY1NVwelX>0$n;7KP8HD{}+s0fx++5GYOy3^!E5q5$ z)M!QVxx{WXF`xRY9t^|GGSlpk`Pn+h3Wu;#nDHVdh5e5gw>JE(ZLR`O2KC)5MQ|eZ zm?jwduZ{2L$X*5qQKJbYP&w%8beK1FSG%ZSol`Q|KcbvQIh0vzzwDit{zKZUbEX@| zV6_NiAYlSE#!t%1^|nJ)pNqIi5Q}LfFZ*taQer$lM)Nu}OBP25(<%Ea{d`U;UKc&7wFeX@beZl@bI0z#XLXz+& zlS9of!wvZ6f`gx9ZEwEtDc)=eKa1bE`Z4heGa^SXAajsD<@BYJY}UTQ7}N6sRO8$5*5-+{%CrI5<^gt!ksnR6e>@_G zOmNk5hODbArFDd^-6wmBMz`_qM-0atJrF-WE;PO0W|kB zMPWlj7vatQixBmySJSwh3+r!DnxN$X}wZ;u+Ku>Q`gqSP)v=s53$y-hEci3k~%e_NH$UWRQ)GWys}@ zK@ZK$s;jGmcg*hkItCuzN07J%2M3oa*Zm24%&C{$9Ss=#2U+FeyIXrk zhg)W*mQ8kb0(SWgm8y!xW_5+`F3ZFE9~N)j-^xI7uI?HjhLRz7yg!njH)KPTE$bk!Cjh)FJ{o@#iA61x^?DA-*lN9d1 zgy@0q2IblLT)bJwzyP?Dhd~`@Z0yMstt3N%&c>4SM@V#(M2YVesJw0E4@ea?$m1I| zjxQ`Y*CjqoR4o)WOpEE;on5EmKBHB;7#NtWNwjS1iocG!&8aL%i8+n(PrSbI8qUM{ z>6C2l^_&?0H!JyDl`2lv86`Tiq?lqP<5$N*_c2E{9`ozR zV~17;c+@~<_02tklyd3URNW$7NUA4Xa3MBWaJA(7gIf>pw`%EE=9NV=7fRzKM;{06 zZ5W8R-)&-I5*S^nBVLB``dN`4Ge(8hGO~2Fd?M$tyG;DDG!P8ofJYUd{{F5$^o$wH zELC}I&q@_W5f9JRSZee2`U8sA_qRlYIffMVQ8;3CO&XBLNN4&c8oLgA!Z`7bSN&n1 zn=x?|Qm9x) zNOt(KBSy`>uQkT7+&N1#BBy{Rgk@#3vOq5;y4u8UNo$ex z6|7YRy@9*&zuLjaTwxn8J%k}hgimN@NVc>lEp$yUuyDJ9Z-lsK?B)Znb(x$D$hTpo<|2psqrkDX@_ErS&KA6_zxDs65=8!>&hmX49Uiv{#Xgq#{fcN)6BD zp~Ee%K9lBL%IS^;p1;CmHKUt4$#V$}(qjrbEtY;|8@O0VSei!#O@%~Ki%&)YJD>7kX=|^8`jSs>8ue0vhPDN zd1`3LwjyN`_jdx`)3%u(`2D1eE1bnTkb6jVRIBbYdR@EZ=-T})KqWkK*jy`A-TqV7 z!_;bB@o%v(%-XdNxM3cT26Y#B0>9RRxBZ z>f|xZE>N%CRSa5@UZ`q>3fi$Wgw~qUiL2GIXVH zh^h5`1(7mNd8w9zbxb3+_N4toO|D1K#ySoo8MR4~ZEsVpdFe&(;kw|Mxt}xLVOE~u zZ~XQPj;9g!^!ZrN$yGBDGH@XU22SPd&fK7673Yo`mjpp2>X1p=q2{1EakHOt|e?StPjd5g(ng zir83q_3EP%AI@Rd{`Wl_ByPmBbu-!xM2-BoazEFH(<(5mW7QZ|`XF!9+%(-XQeC9G zTB^qoxm>KVdV0okFmjV_ZASE1)HiX_9M+CtJH2Bfl3KT@oRoCU{gYMjWl1eoNG;lv z6G!$fts+11OlW0RU!5ps%cnheV{u_&6=)QeL&1$Q+OkVADO#?MeJ|W-;^{@a&*c={ zO#ks9ZP`)bOTqgiKB)-;`*;4C`3`NOc8qQmBNI18oPg2)`A3B*dS!c1PG3EKlJe@0 ztZGTkQUQDN*R=J|&B6GInf1^><6F)NLtj$H(YPD>#^Y4B4p1SWgU;=3-J zWIL4MWA)xE!3(6)opQzUMDnIY1$%O0Kd-M&%2#szx_N%QospEUGq7a7;`c30_`D+G z)jd9W4zKK5D}irPVV%%pt=$Vb#oh{fcaS!%a_L`^VnMVZOF(3LSL5~eTVZu(o4iYm z`S-om=T-e&Y<)w4cey8{dV%fK0u}O5f%SKp8lSocT_NO{JP)WAWiikvMIK_-I~T93 zO5T-9vFh}{KRc09aV$_FUH?4-QLeqETM-I~zMF2am*%-A43_^&uK#U0b>{n_ECIq@ z_3!8X4=fqW3LQ~1C;?TWRBf8|V+~&oeM=ZV3gox#k7|)7P7;u-VdIEBPLDhU>mF38 zO=rVHdL0g4?_G>+ZP+(_bQbK&QwOG70~zauAnTr zMSqF)lHOX_`zB}*LBVCalh1Wf(6WD1mN4eM&wwgUir_lWtu*urF|Uxj-FlWZ9WpUA z(qf1JM12ayK1MnccWE;*H6h`d%kb0sh*i_&uZPSErX&-8rsMi$8^39!tLstb81Oh@ zZ_H3~s>z8;GN8aU(+qaLEHucCts|f#ip5hNqfIjkbs6G(oGHwmj+5=nO9;9`6U|Yu zG&I4~`Ley5lq43|sM4>@A$3zPLe6)v^=^kKYf;iXIAPVEreu8P1p>mcK?qlx%%L{) z&U!5T-0Z_l&AYj~Pwf!vvU&^DEvcm73?#6zWH9t^p&?25gZl1#TX?>f1a z_(!cTObi1p&1_`%AyoveW{ec7JRcwsCBmNwo?zY-12pBAThg`JY()7W)!tV zysj9SqqrQ3D2w|sCS%wxm0OptZe(zXMu?lv<~!3OXL)549lGyK9YyQXS-1Jc01M7| z3)U|qWA#;te1^rZ)deKP4>5fi(Z1>@J{EZ9LMNnpS|MU1v)%<}9<5bhM(19K&Tby& zV3brQpiymBCmPQ~7dvQ^wz`a)kG*WnXkS)!U@KD68WB)PSjG9d9-DA0p`2L%wn2Y% z)gxx==Ule4mEZ|pxz#7;da6s*>j)Vx7F?O{XRt8>cxbVUTDzw7ZKIxe~i5y`f~ z$O2dYxrGKNokEuw8#Qz^+|6S*g}w)iW3E8i5WMV?t3oni^D4MxlZk%vuENhswxvq$ zG{j?*^7x91#H$mcAA^1NYfEE@qO3IR%5OR>U?_D&)6>>qI9%BL{AiUo zU)t|8R2{xVU@KaxOwT^RF&W`>D78LCo{h9FdQWVoAU9N(3+qQXI~B+-r6o6@3Xgn7wo;qD~$Xp2!e^qRJ$yLo}kvKv+6kt+j`RV9gg(XXYT(-NcAiHcz||f&Ef# z*7yV`dW&=l~#9teFr)bnQ=(&)3%;%hUbph%w3C656hP#UP9= z3AYYaSU*-)j2kn=?{Rwj=$RPJoKDj4Md3Ek1;qq-_ZglK_}7!56{%!8JbnuPQDG;_ zdbok^7LHOy_wjQgTm2>$&%a~)&pW2=y}h+Sw-dL7^}U!FO9fnmBSeXAI>JGbY{bKF z>a2CWin0V_3elw|8vm!Y_l$~a+qQ*E5EV%x0tyNu3Y!d)BumaYXA#LcXHWqZ6;Ke6 zP~0-gfT0=Y99BAGRh`6|3f&bB-~4@1qZu?kc`b zlzon8*%e!-9llQ{%_z6VpRH$e=$6Im%`G-oJ$~Xfv}q51JG_SPIMBM2;O_U;SO6z* zsp{&+Sgm|u2(Y4aJ)H+nOWkTbjN25iAZ|ct%N;ZGE$n+L1Pec>&^@6%nuW% zCJW~GNM92Q#awGB)HJj1;-tA=yU*xV+V&KiUtSw>KT>)nMYb^-)l=!dDyAkH!Rp&e zw~lyO>2MTqtBbgeYqXF9Y0>W6FADT0OYSB5ov7siq#A;xR{qsym;d;< zR(gJP&~?`r`*iWj{rM5n!*XhaEv-S@w$ajxZ&&g{7e{c%jZ9x2pG0a;R&2T2c^F;J zeVTvrqyT z&#&gse-urR@~q> z-wd{;g3}27iKG{;LWK8Sa(}WEdN%ng-=V>8X6O<7#-zqV-uR7qS0mh8tN-2HcR8c6 zVh>_!TFvM2rZ#*uQc{LWlA-?EYg)Hnc7?vf<)Y~@&<^RF9Bk|DPt8ofAN^5Ce4n5C zNb@>6>B#-F=GJd0>6vNKSo(Ur$b>1f)P5$9H z67;maz=?aB=2hQUH%-!FB&CPWk%#U9W!!1$gujX(HGS-jv|qR(dN*dR z3h(F$(|j@MNzbLix%AnHMkESqv!&j5EI+TD|5kC(gma@XJU)v&+pMIzF7>q9;127r z?Voyw+3X^3akPz#*%+5z-2J8GIr#Y5`p>7)-@e;OMtllP-}-h-NmcEpvYaf-hwULp zLC*P(<716G`hG%bq(93}(-OV>bVr)+QaZdsjs_#Uj=80p>nVk4lJw-#OvS#SsB?0U zWAae39zW9w$Bi!^)hy}BZYw-Y+_>=eoTq1nKyK@%%iV~bHpy~l;ug(e&M=| z&h&^%%4If>c3An0(Y8YrjG5%ZtC3ftKG=yp&Dj90Ox)9-yTKI$Od1n35l-Qr2&k^& zUziM=WBPldX-Z2=`}kB9mu=6tlSSo5wKidQx$*W5huG!$TyOP0#qQc=2Tftvh1nkQH2;BD300a%Jb+eG0dCflwrao z&+AtCWKDh&H|7vC9g&}H%Fw>|ER!_Gl|YD$MLEL3$PtW|5iCy1Ma%Pj>}>641Esud z?`b7es=3;GT*{w*ItQ84HtUCD-2$Aq+AmDK#IBYd7I@NU7!?<;aFY4)e0wPME>DoQ zLHBKiR&`N|{uXlJQ36S7r_iF_na0V6KBkZSMguF0Eo48hze86u`zHdjV9dGrHT~fz zj)V1TP1VOsY%d*MUR~f2d}9z8?VrjllxMT(&^$L9oE5Vi)?c;}+9LRe>IzYY@ELuj zS6Z25<8D=+WK`?sWEuIFk>kYa!=c1Jj|d7{^Tapk>8J!;SEkN_(?E?i{qp(&+2Kh+~azzJU78?DV_R z8(Rl_GM2_SWDM}$N20+)r$ zZVJ|UNE%$Ts1Fg_S@^a;A(Ix^z{|Dq+QvYyVqu(9Sm3OE#|4+p@HcW|<k+7pqKm&Hq)yehT3FnIjWR|6j;(~a-TbB)LMea<3#?{wzZX9PAi!wt$lG8n(y zzb69gQgIrfl2B1`1l}4HVT?YWo?p;C^0(f8 z9~(2^1m42)%lr55yCzNO=rX@PkB^THy=r1&Wo2b%W_8zAM}IVbK03N--h8J*YnMrA ziIOr{THMrBU45&aEJH2>sJnsV+UGRC)r}T^QX&(luHfVkt}t83gW&n=*RJu^#g1tj z8Bx_X=~=$AT=XSEu)ic!gwvEq9TxD zmz0!z{=5&GKzLiA- z=UllPH9-Nxo-ZAG-Ok3;!C^O^V<$HDS~RWHg>Yi?O!P-~UUDi5bMwrvxb07p#{GUv zVbQ%{k3RhT4^ilf_v#g%$Ys=}0 zM>KtRXlUwtV6u*RxFYE2>9?W|$pM@Oe|697}B%&V6#38{sB|CF|MCW&AiBm;JaE%Nj8_r1;_>P%Ol(LjLja(+{j zRFqssy8KXd^k59L>htKT_wVu3&Kq}kNhJ28^3P-B2Iuo;<&GV+GKln}b5C`ScR78t z*4)ib6_iuxi#J`~(pyt%Xk@ z4XlpSIj+#Z3e@qS1iUWnTj4DQi}-5Vhu+rKR^1Zw zZVMf6ag%4;SZ*^G8#gg)IN3(}S815ID1W$6!21s6wPC$USrI$G{`#h;tsz;gTz@ze z?bLYp``PwR`m-@tT9(WnzwSxAH6>2w=Nln&du3Hcgu6LYF@ET_7(mh7Jr*nu4(kyS zm$kLEm+Jh<3y)2Gqxl0vWz%cX-}d%Ac}P;;YRZbJovHZzd3bbmixhG7Ds6L`(m5jj zot(EF*kFPRR0lLj!R6&O3C`7PZ~_BW7B zmUPV9=aKs5cB4}Cm#jbgAM>}DOmd(a8pJA#jjO%6ffKsQPJ?sacrZ;Z2bA|=Hw2*W z4LZL0`uSz3EyC0v86DjRC-GlHX*T0=aTJvg=9>)!Z+oy}G}g(TTBpbOZLc>BWV&LGgtGyv$NKD z?&9(oeS`1P(h>*|7VU=G+Fs@MT_q+KEdhExJ2UMJIIVG zM-W#oO3KRO8qVcDcXl2B9CV!Q_=h-V(1>)x%=bUr^j94V5aKkl#aC``xg3?UYRzXR zoer*ad36>mP}H#9+aNjjVD>`qhPzfw0XOf5z9;jB!y;@$agn5p;>-76B}doQ`g)w2 zcJc47T$W-U=~Rj!c6yFnI~(f!WmI>^dsQv1Xe3g| zW6eP84s?asMWesCE?zbs{MuxG5nO79*%kcnBE~{H6fXn|T%r)6g0=D8yJuR;;_(yT z<22;0nIG5O5>pp@*VE;l-Q2dfw=+12jCti2f|{O1f26kf1x(X9DH+%*yG6<6VHW`> zc0s=bG$?O?I?f4ZeIpD#MPd5@cNy>QR7qE)c#Imz%gtp|0Z{o4`{m-psL(CgUzLkN zriK=7(0fLQk^H*`)01Bed^Idyk#B6j&qn|9!_OE+vyenTW5*HVA4D~jwgsk16%&1W zy~(+DDt8hpFY&MIl8sRDHJ}3<*?a?v)*LpeX!D6AAYC!PPU8Ub;G>{|vt3$RS{H+V zLLUquM2~{bP<~IIhkNa%uRbC8dKTd)bR&Gu5H~sJD;R0%>FH?|7_`t4Txz@787;H2 zx(Zw9zLg(1a2H#4r~04X15I?&o2TL#YRG#|4?%UT|H&Q`^dOFdON}?o3|Lp-X!=Ux zR*{aNX%qgEti^ZGSwnd6$2etr(Sa8a7xR|RXzB5ucpJPlmB(x#_Jp0o7*X(7M8c)wmgE|%*)FQ zo`fx$!)7{({Q_7Vp0?J!e#@$RXMpxPH*I0d z(|o=LgxxalsZ(mj%Q7Myv=s=amO(Sqf3kQ0!9d4f2a4&C{DElyuA;X|Jyn&Ju;Pj* zr#K-AyBw0f+w!ph^n~i_fp19a>aOuo>2rNr9Z*6#fp!}j7ijT>3B?qmV!@8+MuVA| znQygCwp;Di670JnYgpk(!^HGTNo%mcFrLdQ z0y-hRzQ&2^Y^cn^&0SVhB#L49FLlw|R2B>aY-^PeWyUxROoRT-R zUW#&SxAQgY8ndnT%P25waF7ZuX3xm2-hPqVei>;p;&b9izqN?{31qUM!UKou7n;M_ z7Vcl~zmjE~AeC1iGZlV=g^>@Yf7exPoIkZ|Mrb!%&D zP?5T=j&Sn}JqrpF-1@+OqUnCgxix*yz1iP%gR+kp#B6t!ywp=^eq8IEM$Z5QsyZ=Dy7F#VcsO~8#LmtRj8EI#(2x$zZP*;g zAShX9kuB9i`%l=-KgL2~$rJD5eacdZ4cp6uC7kElO~xvm8vKaYU$VSodC?WudoVeQ zPlNdN9+HTUxuKy1$U;H3V8+cj9Mru+z+ zz}dfrxF6$lujA*x1gu^+Ub6AeaXDGlsp&Cni8|P)_CF3d&dS?He&b!#g=BHlWf0Ut z1%aVB%gUj~HJr_MUselNl1P*cA4Hb6t%P8WRL91~vQ?um`z=kySHf((u@|$R{bnaC zMmkJ|$eqrXp*Csb9eY!(IaQ*;d(y?PYvS3}a>F7T=hxdUmcR0YCIC@~?Xf(N4^>Ng zIy(3GatbG9U{8lSIVq_os0KhHkx+Or5pe1k?Hjm=LM01lQuxBDsBT=tKr;))b@?jC zWm1lUwF9=#bG-X<#t+WI23PQFR*a^)_?IW^ri5MK%gnX?udW-jllSsllAGX5-k@_b z>Fk;2+~*yP*UQltIY*cDM?taw7Ii~vRt=N9(OeGF#58|BJx_+3h5#K^869BnEh*9? zB-4>7)K+8Tr*XUy9x%)MyLAUuw`{#Am{IVUGz2lIX9_|b0*`l>`5!4{{&DBj9N|vW z&d@^tlE=;NXVJ$dkKq0gWBPD7g3jn*>s9Enu)G`}5%F|P0O3dtYYd+MR}B|q(E%~) zCdgpeptej)YX{ooD!ykoSsAh)%(k<$LtlYyFq#R>SCe_!v3^&jO`Q~#mMX}|pjMtJ z(21k3A=biS{xvl>n5yfTn%mvWIj2JTx=yt*;l&(NGgm%n;Ui&}|KC z;CH+rqW`-X?i~4lFG#Z2kcree!vK2cDdNWvReoP}Q+GKyICyzKL7HDymVtFm$Q#J$ zVW{5!dt}kl5suq zB3i+%{A20IkFM^px1^VpM6&H@x@>fi_GHhL z-J0yWZ>-#DaZl39$-mXYDkEWU96P7`A!UuRLm&Cds}S|z^g_1c1WVSxO>4;azxLnFE+=?i+ka&$G1Gfh`n>c1+U}}v>QKT<6Vir)05xmckRho z?swTY+(eFDzk3w?`6cjaj<25?e{*f=s?L{bCa)iDWWJHn5r)EvZ_{?8&4UELei~D> z8&O+$gTd*A_Rb+Zlc`pTyN*u^i2gAO5B~U2p(hy-dRB@mnz-vF`dBu>!SY24iqlD- zg1Dij$N5T}9()2`TRV<{i|+iq%ElEudAx<3R}CaJGwPCqnK-KQ--*LsZL88plz$xB ztR5)w#L;JFe%fQLIhUjQYD|)hER|gR@<~QwVAn&wMiIvRm5z_g>od%Y*P_I9BQq*T zzw+IxVlv!Dx|O^a6mXTFNA6o{u@xkLW?nTsZ9(S=e89BZ`N!3SW_{lf%mQ z*642Wk=}~M&01**vZ|VDS_V#(J~oJ&K*Vej{36{ANhF&I;ufm)jPF z4x&GjF3a8Pc$iQ%WHJ9pKMW-09E#h-RXglrf?E#TPZ*qNYPg=isW{e8H<0>1Rvmeo zSSc#k84+YVd2K-}{ta@n&SUW5LhF2;PzQ2w>qzm%nQl>Y4{>Rj+O2O^gyr+e`t?_4ms`|dA zTwsWmE;$yT!Z#^jOZ304t!a>p0*1ZZXW!e5TkpI(`T-=m z@k-cV-OylY?oYzgw6iv;cQU(3`*tF3&y zrK30EoRA-U$GD$;uHtgkK!uGEX%5F*<5xs06)qm%Dk(Nvp#tk#*PMQBm##7aa{7jS<#>{QjR6Z_AZR?~>ihzukTG)$aPwjz1^J zugCuaj6`_KpgGT%S~@BGc>R+_UMx~>eJocwt!CwcrAj*JYu;%b8auuXl)hiL4 z4wxl$kTmRRsWvu#Zzh zW(Q@Wyu7>MA9^wPsoq`-yn-m?WFn}m!B4!>HgI|HgOO`|sh*weOSv`3 zxk_$MQTq1ATcbz85!r!vy}}F2-1Uiln=ux!RZJzW3}%1m|^J{bKnK6Wqs6 zfQ3XY@=Byy-ljKw zb?I%t{3`pMmkb{wl!C&ANlAY)>3-zNaho9YGEqy`Iw1%zi`{raAj@}rotnKeJ-yK*&lVza3xaVwxTqkg zD6lWw|1exB_jN|yxg$qIE=q2LO?Rn4rG<0Eio5ilJkRw%v{X7@{zMz5P>ta7fB$1w zg{O>*_R+=u8(8WZu%!=4gl!Eat7V+CZVuwUq4u;iUT`9#|HaQFPFgwH&a2XQ^jmf= z>Ioe;Kf9L4)UI&7k|`})tk)W>d{? zce+|xezI);WwiCeO=vL>1%_f6(OBAku=0jrKS?|+N;8oEY(H;#&Xe<9{A zR%PO&NUG@02#EaHh`JdYIK8cize}#D7=nX9(ij_9<{JBM;22+fggL5vd${*JQ<0aD z_;#}~nT3S~dPqU7tBjeRgtPNqXikw0=F~4lf-0duuF>kzY>m%j>E|7|p&dj34ec_X zv$L=OkvkZ6%9$MmIYQ&60YXGLDI9MhzXv3dh?3_=OG^mp^&2;uUtcqQ_)rW3WA}uV zW@n?IQx|vu0Ped7rDev@&ItNi4~&e&C4quXum|c5;baD=g4AFJI;M3*oGaTvSGT3W z5L#hULk%9FCbz<(2xdzEBMrz6snDCRgSE{mGx@kggBEO|s|D!(2)LP8}41>h`Z z;}KIFJa!|cPyyca+4~I!?rl~9co1+1dHGiu@%K6Ii#VJFjD7w6-Tg9ha&lHyIj}S} z7?GS7VXt1bL4xy~h`JyVAbG(aROsUZF}V9qDUWVr#)Y}LqRAhsH2nN=!qvql)Oh5q zP*yDHyZ7-RR{Sc4PpEW1tGs;gO8VB&*7L=q*~I*9B)WG%`-W{_>*X~l2vbWO9Lzy`x~#3*Kr?x|E&=CAFZmR;aCLCV_Ov+7$k>_|C$94@x7 zTK{99#$i&pJkkos@^fEL8%d_WMLA8L+_@UlGgQiO>aHnx&R<$y4(;c|Vb6rZi?k&G zA=1?r!Nm80sVS%JRbi;Z7bGhU5X@3eQQ12<=(%&}Lr>#Uo^A>1X=njLx2-E^mmTpq zrNi!v-6mrvfU@2qga=QU*w~VI9TI`;ImlY^7Qq`y-+o1Nz;rnTnwAM=sL47M!5RYy z2jCKB{a`2b#_Rc(dGR*6?r(*<4+(Y_3w9Z}v*UMSepE}7Ls%QRUY-op0vKZs_VpRm z`Xh7u_EtYlsyzpOOs&^&TtrC750*1fLt;%4a6utdB6OA;NGLyssU{?uY&nQ4(wHjQ z<@U_|_8%M^-LRbKx)bOX<0M$%#F-`}%YuuGTdItOg_Zqwl_T2ZzK`e#5)ThA7%6Mn z85A5$uwg@SRqgflaMz(8ToXXvR{A9|u(%YPd7$)jrx7;2$~_+c$7tLeSguaBV7gp0 zfnABtB0}^m64{C_=tWs8G(U6F~bq$-D)+ zPD8neJ>eD+4^y}-}H=k#IbSFdi+{;UkI6@XO=UHmRpnTX)M3=h|4jOtr}@o!*zhGL7_ z+S)2K_}sw18^WH%-`@ESz~JgHU$W&w!T5x06d+n_eu}Bp)%L>55L~+pc7!4+;XvLo z<%v`;D-YarkrH}nD00*ni7B334gq(U-kQ`87Y7PLSNUn=l#Um50f!XJYi)10+xXd- z_ao2Xvoo+Ec?wD9i|gm9ed~6vuKqD%SlS!(Ceqykx!w<0j-0ZE&4ruE@+K#jdM-(I z0~oz{bf3+6y0PhR*nWgAch)8d~vJ^RaOu~E##D<;UIjnSIgoNHSNy1#ISpX6e* zsxs<(#_?A|1>}ozVkxDPJTpL_`9*z^B=Z36mY#~DKL;c4TupJ2m&eQRg!Q8n#WVc< z&2pScBu3V|LT^i`@+~}>; ziH*mUv57ATwcWMatC{+yeD(2LT7~9OUBk+Ou{^c&aUN!;Umn;fX{8h8W3xQf!#dp@ zKMoh|t-48hsrdpe+49oT=H@0!J@QgvY2OjE3=rA@OX5OSxy8%8wD+iM_IjT5=@^jA<%(up28ds49kR-7zykGIQM~V2Sl$vF>)EO ztY(1!3%2z?JtzDftr{Zng6kVs zXvPNEJ{`r7G-__NWfn_x0{^mlVgCFi0ncEOvVntjzLiQpPxfUc!wbk=W<0y_>m~>F zKgnxqb#{684Y9~}b%9s#zG@O<))u`%QhE74ka-MOUcxXg%+GIzM0 zKQ}2ZMDE8@$aS`+a>6uMV~j$lI2A}G(&3V~=krcyO_<^q?`v{!6n*;C+0ikdb**tA zv=EJDxZEZE6GHk$aRcIzXEjTUo>#&cmK;%CDbdkn-Wf%**nR8R64W=a;HzKuAA4|? z1(w`fB{NJ-e5tByUvE+8!*8xUAVI*Q&C;E*)U-0*h%y?gaH2(KTUqq5X5i3VCg>7@ zuoETc_VZf^kc$rEmD`wVRYgTGs$T*`3P65L*U8Hv-yUv2B+8~o(l9YHQczOv>jKha zYPG%8?^^Fw@mzpl$N5WRVDSHr!!Y5yzRAVAYL*thZl`sgS8>y_oVmWqXg0zeWlB|@ zvqj3%Fxc_Bo;pz(=1lpAk00L%^Y-{&*iMi7yf94<9ZX^POSp;A^Phe}p%XeKrVvdi z1hoGXmZtu+Wj|mCVZK1% zN?{Z#DlU#ty1>=HZ7=fp>Wf6B5=d^k`Cgug`et~1sfoztlVV*=qA)@s(tov ziuK=TP2`yv7)z{r>7uElCz!`8EfTt@UB^ghjaQ3BU##%%Dg4gj7yW*qE9ZF^tMU03 z-vcM&my02Lp$UAmjmk7zz&@U0<@0I!XtK1tyw`;up|{(Y4M#4Ga3QA?N*8>*GKwL1 z!pK%-;vgPi;%&ad`C|7a{AfmGRH$j^9FzAs&&Q}Va$g6+OVNivF+C)}%R@KykzOPa zUjfiWb$93dr4b0G``(?FX_mdEoz*L~UGXHzkahwKXQkcC7wLfYc!F5X5$e?7xg6~8 zw<84xTU(bJg7I{9bzO@q-B(Az+}79E7mCg~eD@r^UxJVU5O)lQE2xELdsrs&;ibf_ zRVI$g9n;-v>h!h_9~_LvLfG-f;aTG)HTA+g8~Yiiz`Nir=WRIWjD&> zMlX(9jh?SnU<%oSd;wJmWyYE_C_4dR;fe0C=!K%Kzc zizjUEtWJqJ!6=O_a*-s@k%VW)veyzj*ykPY>6JijM&MIiyT6i2w4&7xu&$cSdOn&7 z&OwGInX#GD+z9IEuE!c~2q&Ahbib*;QSO^ zFa8WCAXK8qr{z2BwpZ&*`A zpQzrttWk)B&Vb~lQ!;TV>X_4*8Vkh|`JdPzFddMLL^*(R;*IV-KEBV{*)K+PWxT)4 zOXt97jJ@r7EToq+3I#gJ`!nn%!0wV|WN4c@Iy&0gGT*>kq0l=-C7%d*tO+{LUWjiA z!X?S3kFOQQ=DR;6HvII0Py4RxW7GqBR&b5f>K~9E}$FW>iHmZEuMj9h@F0>n;m~IP{bp ztXs@36!;xa7@ViX8%Uz4^|c5a^&AB@k3ONYj?OA(m;|TP2lhT=@5MXEL*Lp5MxO-i zhh4M^@b;R$u>HWIA$s%hnli3wg4O(VOR(-+;T7Sakq_^V#oc~Q_?8Rj{RYbg9l^4! z(tA4>@PNq3gM-l0vN9k8Egn5e5cKL99_G7wa|)bJELwQqhlLR?nda7GjDE%x8#Uct zD$zb6rd1`L`7Pu2k1RyvNWobgby1o1N)HoW_`)saKAvpLr9=@0a^;iI-Z*^ggwyFk zlgnQxxegRg+P5>cmIlI~wEUTPKmUlzS#OQd_5z!1on-+ArXFb-Y~Xn3lH!fJ!(E9Y zKow~HtiFEc4_@Aw0E`~wzA-8td3FXWU6UYEJ{Elnf9U}Nv+MtP3A(=d56+bcD)Rpm4~2u- zivGIT|G?P!{dGMS{aY!D`4Rrb$c_%BnhS`ojH05qYy?C9a%~K4kc!(+d})dK>r<%s zfH|I(_qP6vxFvTR3}|6lSC+uUBKE0n!uz|28hHPBIr}?JgZ$K5DJu3W?^`xW_LU-;icx_{pbd&R$iR*jX_9W4L!8vsQCsHh09-+L1kC20nJ zL@`F*vbQhC#>d;tgJr?5uHOvefkKYPvuDrZP);5mY;0@`0^4})e0T3UKjvJle)sN* ze?K=&|2LHNZKsozrXlN*4 zhrnF7x3@z$K>rG{n&dR#q*_Yv8phD}BfKb1k5VATJ6G3><#e)e=epDX7Bv@;OpAc^AN5A(NM8^g}{q&Muk29e-Niewmd?6z%e9G4o6zLnbQKl znduqfZvolrg~S1;KV%k)7KnQHTZHtN1s1dN3;Bn1RQNXfg3ryd3`}SjR)*UhjIIYu zlG{apNm3nAKkNIw=S267zjc^mJWr9V#9sF$zk$u29aaVgC-CJWlv7M4TBT_LomF2F z&WlAhAr4{T#C8eGN16|37q-x!S%GA~cTdM6n~o)Bzy*}TzBdr(>b`GeN7e_1`B|rL z0XB)a5syhN|CL0wyW>M|8#PxKNn~Usc-HX&)>I=K*g?cKx;x<$UI-2?uirckEL=_@ zgk=aOxTc^HsdyuaudM`255F(DvWJZW1 zrEWfq_=bN{hS|0moH!Q$%HE+sE>`|`lu zN57>>7XL30w$!L7Qjxxo{}iyo7*;r@{E~OaXjh3%&>Y7Z&fwSz-%n>jn0Q`YTRSEp z;W{NH80koATEBkrB6%2`p)`7n;$dlaFJ0R|djU~h1Qur}dxMvr(NO+|zUa@IRB(`j z1aZG#+qlR&$&yIld?6a6M*3ccM2>@W*e6nB0S-~z+}se_rvQiwO#3&$D|Hm)B26It zb)Js>g<@*gy-(}rDtDpq0A9H>(-$*M=i z8F$6MfA0y(E_=TWp_iYb20}o<2Qb<^xu$UH^~FW6#hyDbpLXN^)~YUEj@?ct!9kuG za^qYc&>s03tvg@!U! z5RuswnsT~v_-#9L zAnu%|T_EFhUob}B+qLtZIFp*$fBNv1TRZtH@sJT5pgd5f1ykr|6Ea#XIy)NL5n-zKL1mZyEqy2Yh(iP zX``Gg=SyXa+X~5AN1Cqk&!3OlW7cARzN7@nUk`cnhFh4=fJxtLipKn`D;GCl4pv(u9HsjlOa-?G1yCjg7>_#FbqV zSQ>Llj%*l~!>K*`PQ~&q6!YxdoTjdBPQX+N$B2jGPw+vebd`F@a1S6WC^fkUo&5Mu zVMjOyQ^t2~bFQl3v<>^h{LBnzz{wGuOVx+R#>PNx)gUIc22+1(+RXv0i&T2@*G6m^ z&9Zh)!Awji@7|A8!iKMGd!a9gg!xX=>Q8E^!Zmhr`5hA-*pv5X13nzO8mG>OyiM|m zXQ)M)1svyuI&^Q^Xr+fM&u$p+<;G>!J|VCY4sZ^zmH72kApTN(Z~x*-Vu950^MSm* z)yq^{0jEt(L2OK~gYJzettTf&jA!ugCUlJ^jf4w{Y*trLdw6`Xwki1D`Fe6M)3s)v z!B(+@_ZLZ3_m6MVm%Y?{*xVQtle5-Y=WGz+IL9{&)ABRMvky0_Wb{g_w=YPyf43=q zU=zlEfApX|AmNXKhpXaeUd{U+u6`QnPQb*qx&o-A@<}@E|E|1fXlMwFW7y$W?C)#| z*qVmHvX8b1i1qv%nP8>F%3iI&oA)&%R04(DMZu{_5r!u+1Hx#i?&4{-0gmRDNXmrvBF z4c?vRO_S1nuhu`|EzxZr(!ste6i^YlkA9*L#<37;q@iDFu5BM@l}WP9ZGex6Z>if} zW+K@sXlbHTU-XW!eb_9HTbTFNDct(4@hp;%gOHUsO}6GUtd8a30HMr^E_myLG|C6-}r&; zEKijA!`m6p0NP8HeD#;SI8o0-fU5x%-cW|+dhSw9acY*dgv4#(+q`aP9zTBy`HW@~ zqzZtBoIn2@=8<}ql3|s5s8To$4Goy3;xFIV`+dx-G)*cCj$iGaV>-+e`o^ZaE{dqk zM{NZX;00>6?;J5~KkD35pghZBln7Is+os*;{q14Oce4^dea_MKbTXpB9e{#FqK6UeG(9m_XkURnjJ;dPj+W zs1>jue!X+p=IPOA^l72bufZ`Zxxu5X>M4anwyYMz{>9z|yFH_4{k}CEE0J5BCr+Ik zc|z;?xCZQz1GD*V`n}=QVd!k7R|YnY*?TKqTc6d70$8r1FHzQYE&TsM)C>#^P-z;8 z^MlKe+T*w(k%v8*E93;=XjVwf73GwaSQ!~p-N_=%O(Q|*F&t9~JMh0$SaVs>tlGi{ zK*^4>!`D96$QDa8l*`R!W##lNJP%du$4^j;p6N48!oo(kFBge-LVFL^7*kWz;5S^9 zuN7AN#EwC74c_0xJdwC0bSla`0f8XqsPI4bc5h1Ndriq(uq(}#&yLIlX+=}Cq;ggh zP#*3-u@G19LFv+mk)ffk{g@MovZJc ztlIeFwbu@N^woWwdzOGsGk9!4y>?)B(({slDM8+&DZ_YL|3J)X=Ss8JzQVO2$Dn-@F}GUP;K zrZh&<+lBB90wsibOc>Spy?bBwe!hZ+tb$RPMXnu6pIEWwBk=nn`AG@e5b0tQ-v98m z-o0%>u+u3`(Be>ovL)W+d$z88w1D&E?6Eu>mfaeI?zD!>RE5$GZkRyQMhkBk(~;2t zZZd_1keRH;)Lx0e$?n#YF#XSuF5Y1?a{fH*3O-XK{rbEcyyw;)xSu?-&hf12yc*`? zo|$CYp=vs;=$*g$D$Uuxbr~<~+7Uaq@5B2qh<1Ay_Rlq-l}}ii8zKcNO#_lnsu4sV z{wY{Li^sqIU*+l?z6HCnNAb&w`F;7+*pbS=E8vJsy45G6?i6nJb$l!u@KA%q;LW)M zIm0#)-s+!SQB>X^kwQMveuD=WUcGoR^$Al7G-cHFKdaSElFy)wYY#Umn+FAxG-IR{ zcVqJfy^pGDW-n3+JB?fq6L8ZK5oo#)Gx$_!#I;1Zl$7c;&%1Sbi$2$(rHeeev$l-& zd(MNlu)G>6XXKIAdA`#trvsx^%0`QAncHI9r7&q%iQF_ zpF9&gkSrp{T30qe4HSG~9Z*(_(7t`4MHh5TqRgs|A3g4PkgjUlylnMd=b*)fQ||b7 zgTwkJVVg8E6GlKG{wT#;=9EbcIPuysT&87)gZ88z;&<(LSC`63+a00HArdlYp-MR$ zg(WV4bN-sJ?;f;YHwF?9Os%1JdgZs&t;>G{t?YFOSGSI$UZuj0*{}6HoC}>ZO8$P8 z$|fW90i)`7L+KaTj4WA24l)KFj~OdxWV+zR)b^NO=)bU)6Y$z#<=tuR27SP^zIV%! z$iT{Q)%?D(8ZVAK_5|O1kvHtFev;F%V&1bB8Ec->mA(txkMpU|zMi4KIP$VaLHl4& zVL{qU8SUMkGrx6|xMj1e79TO-8_|2U#=0)jeM!fY$g~UTmYG&yW~19rc@I6Jnblw1(~Q-g`7`tOSv{^_dfc^3ZpGA$Kl|WMa$h``bw$Lh%u&Wnm{SY0N$+PscaP`b|2u z!ArENju)OBTy=)AP^}L&zWVy3BO_{0UQ-E;{P3+Bflhb}x%41^MOE8j?Qj zEuUUrS*^W_KnresBdM>i2QPI%wcw&6f|L+SoFIeHz~GV`4}+ir6rii%bHBf12)_#N zk^;`jZ&h;E$@%2bm(D#=B-m*lO#x%a#5|K<=Ib_$#&ZkuT~Bvh=YC9S`%v}S3b#9i z9FX~Dh~)J8F%zLV$gZ!Fx^(iU1vLA8UUbNc--{M@iCf|e{$Q`R*?)5F&Iu-Z((h9J zvBHz|@@U3_Z-ZjI`e5U*zSC*)Xy#STgji9$#lra>g`Gv_BKgC8bozBA!O3Qs2+O-; zd{dnPt~HyozUGolQ2O@Q_27Ojc31mk!oAkdU*>roMoJgQsSB&;Er#6%*X^yIjs~!d zQInnMmRV>7C3U}8kT@0TDB$}z_qFpw65at)t1|qgWFzE7T&GzRb!}g~Pe+2Hh->-h zL(9pf)FZVuFZ$z^cE7FJb*=nZwaCgD&U`=3#7M0UoG+Cs9)(J~8}Nw?|U@S+-%( z*GMbd`eKnelWH&dlkc^TSMEfcqn!*#raSj5Rr{0jHt}@Ioet?2vo3CxsO^5MDe@h) zTDW6#bS@fS!tTh%M(dtUhK4=UyBpc_WZp{LZ-R#x%|w*fY)4wQe5;LrBpuF#<%@Pm zZa5qpRf?8K24{txxcNM)Tb~?Kz3FYUaOfZAt2{%;8g`K?fPdAI=1=?L-`}aLw-e#I z9m^*FImEx0Aj>}Su0=3O%+{Po?WSAk)^k(Z6X?Uf@I4<_dW&fVJcWh!!jIo>d6TH| zB+=u2RB2Q*JSh%s`A+}~=O)SXbDxBvUmARYo~*+PB>QOU)_vzc+` z@Sxd#QgS=uQ=5!{&&>XJ9}TCnR;SlRX9c)x`}j`XnT;{Q0CRb2X#9gRIu3i;XLx$& z(-EEV;;WO%pi6(g0P}C%1&kdGHq}0HmoM%=emwOy5J<+quU{yPE)57Ns((7Wcs9U~ z{=PgV{^oKpKl>NFQeE1)oYlW>48QW!gap8`DOFQ@bNA&PyBe&&K8gYiJ$ZOJXP9>J zfPedyeT}x{-XnDCZROIpSOiHjqS?=%4^T#;6s%Sqmx2%0&2e)QX_-_4pC6KVDrnJTs z9}5GHqS&^BF~fYw;hRxQ5g!-tP+mG-Y4-481BNpGeF-XogHC(<52?^nim9Q6%H`VG z=*d|oKMX;5^_L|E0sQltNW6U)R4&#=-0VYmMCh4LAAETln|+zyI4mS?6>r Z!uIq@ :information_source: This plugin requires `v4.7.1` or newer of Nebula Logger's unlocked package + +[![Install Unlocked Package Plugin](../.images/btn-install-unlocked-package-plugin-sandbox.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015lgGQAQ) + +Adds the ability to create & deploy advanced, configurable rules for setting the retention date of `Log__c` records, using custom metadata types `LogRetentionRule__mdt` and `LogRetentionRuleCondition__mdt`. + +--- + +## What's Included + +This plugin includes some add-on metadata for Nebula Logger to support the Slack integration + +1. Plugin configuration details stored in Logger's CMDT objects `LoggerPlugin__mdt.LogRetentionRules` +2. New custom metadata types + - `LogRetentionRule__mdt` - Used to configure rules that set the value of `Log__c`.`LogRetentionDate__c`. Each rules consists of 1 or more conditions, stored in `LogRetentionRuleCondition__mdt`. + - `LogRetentionRuleCondition__mdt` - Used to configure field-level conditions for retention rules - each condition checks a `LogEntry__c` or `Log__c` field for a specific value, regular expression (regex), or field comparisons. +3. Apex class `LogRetentionRulesPlugin` (and corresponding tests in `LogRetentionRulesPlugin_Tests`). The plugin class reads the configured rules & conditions and automatically sets the field `Log__c.LogRetentionDate__c` when new `Log__c` records are inserted. + +--- + +## Installation Steps + +After installing the Nebula Logger's unlocked package, you can then install the Log Retention Rules plugin unlocked package in a sandbox. + +This plugin is currently in beta and cannot be installed in production - however, if you want to use it in your production org, you can deploy the metadata to your org using your preferred deployment tool (changes sets, sfdx, the metadata API, etc.). + +### Quick Start: Adding Log Retention Rules and Conditions + +After installing the plugin, you can add rules using the custom metadata types `LogRetentionRule__mdt` and `LogRetentionRuleCondition__mdt`. Multiple rules can be configured in your orgs, as shown below: + +![Log Retention Rules plugin: Example Rule](./content/example-log-retention-rules-list-view.png) + +In this example rule, for every inserted `LogEntry__c` record, 2 conditions are checked: + +1. Does the entry's parent `Log__c` record have 1 or more error? (based on the roll-up summary field `Log__c.TotalERRORLogEntries__c`) +2. Does the entry's parent `Log__c` record have the scenario 'feature B'? (based on the text field `Log__c.Scenario__c`) + +If any `LogEntry__c` record is inserted that meets these 2 conditions, then the parent `Log__c` record will have the field `LogRetentionDate__c` set to `System.today() + 30`. + +![Log Retention Rules plugin: Example Rule](./content/example-log-retention-rule-with-conditions.png) diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls new file mode 100644 index 000000000..5e52ba96a --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls @@ -0,0 +1,469 @@ +//-----------------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// The core functionality of this plugin's code originated in https://github.com/jongpie/ApexValidationRules // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//-----------------------------------------------------------------------------------------------------------// + +/** + * @group Plugins + * @description Evaluates an `SObject` record to see if it matches the condition(s) defined in `LogRetentionRule__mdt` and its related + * list of related `LogRetentionRuleCondition__mdt` records. The result is then used in `LogRententionRulesPlugin` to override + * the value of `Log__c.LogRetentionDate__c` for any records that match the filters. + * @see LogRententionRulesPlugin + */ +public without sharing class LogRetentionFilter { + private final SObject record; + private final LogRetentionRule__mdt rule; + private final List ruleFilterConditions; + + /** + * @description Creates a new instance of `LogRetentionFilter`, used to determine if an `SObject` record + * meets the criteria of the provided rule & filter conditions + * @param record The `SObject` record to compare + * @param rule The filter rule to apply to the `SObject` record, stored as an instance of `LogRetentionRule__mdt` + * @param ruleFilterConditions The list of `LogRetentionRuleCondition__mdt` for the rule that should be applied to the `SObject` record + */ + public LogRetentionFilter(SObject record, LogRetentionRule__mdt rule, List ruleFilterConditions) { + this.record = record; + this.rule = rule; + this.ruleFilterConditions = ruleFilterConditions; + } + + /** + * @description Returns an instance of `FilterResult` that provides information regarding + * if the `SObject` record matches the provided rule & filter conditions + * @return The new instance of `FilterResult` + */ + public FilterResult getFilterResult() { + return new FilterResult(this.record, this.rule, this.ruleFilterConditions); + } + + /** + * @description Inner class used to handle determining if the provided `SObject` record meets + * the criteria for the provided `LogRetentionRule__mdt` rule and associated `List` ruleFilterConditions + */ + public class FilterResult { + private List conditions; + private String conditionsLogic; + private String conditionsLogicType; + private Boolean matchesFilter; + private SObject record; + private LogRetentionRule__mdt rule; + private List ruleFilterConditions; + + private FilterResult(SObject record, LogRetentionRule__mdt rule, List ruleFilterConditions) { + this.record = record; + this.rule = rule; + this.ruleFilterConditions = ruleFilterConditions; + this.conditions = new List(); + this.conditionsLogic = this.getRuleFilterConditionsLogic(); + this.conditionsLogicType = rule.ConditionLogicType__c; + + this.process(); + } + + /** + * @description Returns the `LogRetentionRule__mdt` rule that was used to create the instance of `FilterResult` + * @return The instance of `LogRetentionRule__mdt` + */ + public LogRetentionRule__mdt getRule() { + return this.rule; + } + + /** + * @description Returns the `SObject` record that was used to create the instance of `FilterResult` + * @return The instance of `SObject` + */ + public SObject getRecord() { + return this.record; + } + + /** + * @description Indicates if the provided `SObject` record matches all of the criteria defined in + * the `LogRetentionRule__mdt` rule and associated `List` ruleFilterConditions + * @return The `Boolean` value that indicates if the `SObject` record matches the provided rule & conditions (`true`) + * or if the record does not match the rule & conditions (`false`) + */ + public Boolean matchesFilter() { + return this.matchesFilter; + } + + private void process() { + List booleanValues = new List(); + for (LogRetentionRuleCondition__mdt filterCondition : this.ruleFilterConditions) { + FilterConditionResult filterConditionResult = new FilterConditionResult(this.record, filterCondition); + this.conditions.add(filterConditionResult.getCondition()); + booleanValues.add(String.valueOf(filterConditionResult.matchesFilter)); + } + + String parsedConditionsLogic = String.format(this.getRuleFilterConditionsLogic(), booleanValues); + + this.matchesFilter = new BooleanExpression().evaluate(parsedConditionsLogic); + } + + @SuppressWarnings('PMD.CyclomaticComplexity') + private String getRuleFilterConditionsLogic() { + String conditionsLogic = String.isBlank(this.rule.CustomConditionLogic__c) ? '' : this.rule.CustomConditionLogic__c; + + if (this.rule.ConditionLogicType__c != 'Custom') { + List standardLogicPieces = new List(); + for (Integer i = 0; i < this.ruleFilterConditions.size(); i++) { + standardLogicPieces.add(String.valueOf(i + 1)); + } + conditionsLogic = '(' + String.join(standardLogicPieces, ' ' + this.rule.ConditionLogicType__c + ' ') + ')'; + } + + List parsedCharacters = new List(); + Boolean hasFoundNumbers = false; + String foundNumberString = ''; + + for (String character : conditionsLogic.split('')) { + if (!character.isNumeric() && !hasFoundNumbers) { + parsedCharacters.add(character); + } else if (!character.isNumeric() && hasFoundNumbers) { + hasFoundNumbers = false; + Integer foundNumber = Integer.valueOf(foundNumberString) - 1; + + parsedCharacters.add('{' + foundNumber + '}'); + foundNumberString = ''; + parsedCharacters.add(character); + } else if (character.isNumeric()) { + hasFoundNumbers = true; + foundNumberString += character; + } else if (hasFoundNumbers && !character.isNumeric() && !String.isBlank(foundNumberString)) { + Integer foundNumber = Integer.valueOf(foundNumberString) - 1; + + parsedCharacters.add('{' + foundNumber + '}'); + foundNumberString = ''; + } else { + parsedCharacters.add(character); + } + } + return String.join(parsedCharacters, '').toUpperCase(); + } + } + + @SuppressWarnings('PMD.ApexDoc') + @TestVisible + private class FilterConditionResult { + private Boolean matchesFilter; + private SObject record; + private LogRetentionRuleCondition__mdt filterCondition; + private Schema.SObjectType sobjectType; + + private FilterConditionResult(SObject record, LogRetentionRuleCondition__mdt filterCondition) { + this.sobjectType = record.getSObjectType(); + this.record = record; + this.filterCondition = filterCondition; + + this.matchesFilter = this.matchesFilter(); + } + + public String getCondition() { + return this.filterCondition.FieldPath__c + + ' ' + + this.filterCondition.Operator__c + + ' ' + + this.getComparisonValue() + + ' (' + + this.filterCondition.ValueType__c + + ')'; + } + + public Boolean matchesFilter() { + // TODO In a future release, it'd be great to switch to using a new private enum for ValueType__c values + if (this.filterCondition.ValueType__c == 'RegEx') { + return this.matchesRegEx(); + } + + Schema.SObjectField field = new FieldPath(this.sobjectType, this.filterCondition.FieldPath__c).getField(); + + Object recordFieldValue = this.getFieldValue(); + Object comparisonValue = this.getComparisonValue(); + + switch on field.getDescribe().getSoapType() { + when DATETIME { + return this.compareDatetime((Datetime) recordFieldValue, this.getAsDatetime(comparisonValue)); + } + when DOUBLE, INTEGER { + return this.compareDecimal((Decimal) recordFieldValue, this.getAsDecimal(comparisonValue)); + } + when STRING, ID { + return this.compareString((String) recordFieldValue, String.valueOf(comparisonValue)); + } + when else { + throw new IllegalArgumentException('Could not process field path: ' + this.filterCondition.FieldPath__c); + } + } + } + + private Boolean matchesRegEx() { + Pattern pattern = Pattern.compile(this.filterCondition.Value__c); + return pattern.matcher(String.valueOf(this.getFieldValue())).matches(); + } + + private Object getFieldValue() { + return new FieldPath(this.sobjectType, this.filterCondition.FieldPath__c).getValue(this.record); + } + + private Object getComparisonValue() { + switch on this.filterCondition.ValueType__c { + when 'Field' { + return new FieldPath(this.sobjectType, this.filterCondition.Value__c).getValue(this.record); + } + when 'RegEx' { + return this.filterCondition.Value__c; + } + when 'Value' { + return this.filterCondition.Value__c; + } + when else { + throw new IllegalArgumentException('Unknown Value Type, cannot parse comparison value'); + } + } + } + + // Helper methods for dealing with converting field values & strings + // (stored in CMDT) to the appropriate data type + private Datetime getAsDatetime(Object datetimeValue) { + if (datetimeValue == null) { + return null; + } else if (datetimeValue instanceof Datetime) { + return (Datetime) datetimeValue; + } else { + String datetimeString = (String) datetimeValue; + return (Datetime) JSON.deserialize(datetimeString, Datetime.class); + } + } + + private Decimal getAsDecimal(Object decimalValue) { + if (decimalValue == null) { + return null; + } else if (decimalValue instanceof Decimal) { + return (Decimal) decimalValue; + } else { + String decimalString = (String) decimalValue; + return (Decimal) JSON.deserialize(decimalString, Decimal.class); + } + } + + // In Apex, you can't use comparison operators on instances of Object, so several private methods are used for each data type + // Example of what you can't do in Apex: + // Object today = System.today(); + // Object yesterday = System.today().addDays(-1); + // System.assert(today > yesterday); // This line cannot execute since it's comparing Object + private Boolean compareDatetime(Datetime recordFieldValue, Datetime comparisonValue) { + // TODO In a future release, it'd be great to switch (ha!) these hardcoded strings to instead use + // a (new) private enum + switch on this.filterCondition.Operator__c { + when 'EQUAL_TO' { + return recordFieldValue == comparisonValue; + } + when 'NOT_EQUAL_TO' { + return recordFieldValue != comparisonValue; + } + when 'LESS_THAN' { + return recordFieldValue < comparisonValue; + } + when 'LESS_THAN_OR_EQUAL_TO' { + return recordFieldValue <= comparisonValue; + } + when 'GREATER_THAN' { + return recordFieldValue > comparisonValue; + } + when 'GREATER_THAN_OR_EQUAL_TO' { + return recordFieldValue >= comparisonValue; + } + when else { + throw new IllegalArgumentException('Unsupported operator for Datetime: ' + this.filterCondition.Operator__c); + } + } + } + + private Boolean compareDecimal(Decimal recordFieldValue, Decimal comparisonValue) { + // TODO In a future release, it'd be great to switch (ha!) these hardcoded strings to instead use + // a (new) private enum + switch on this.filterCondition.Operator__c { + when 'EQUAL_TO' { + return recordFieldValue == comparisonValue; + } + when 'NOT_EQUAL_TO' { + return recordFieldValue != comparisonValue; + } + when 'LESS_THAN' { + return recordFieldValue < comparisonValue; + } + when 'LESS_THAN_OR_EQUAL_TO' { + return recordFieldValue <= comparisonValue; + } + when 'GREATER_THAN' { + return recordFieldValue > comparisonValue; + } + when 'GREATER_THAN_OR_EQUAL_TO' { + return recordFieldValue >= comparisonValue; + } + when else { + throw new IllegalArgumentException('Unsupported operator for Decimal: ' + this.filterCondition.Operator__c); + } + } + } + + private Boolean compareString(String recordFieldValue, String comparisonValue) { + // TODO In a future release, it'd be great to switch (ha!) these hardcoded strings to instead use + // a (new) private enum + switch on this.filterCondition.Operator__c { + when 'EQUAL_TO' { + return recordFieldValue == comparisonValue; + } + when 'NOT_EQUAL_TO' { + return recordFieldValue != comparisonValue; + } + when 'STARTS_WITH' { + return recordFieldValue.startsWith(comparisonValue); + } + when 'CONTAINS' { + return recordFieldValue.contains(comparisonValue); + } + when 'ENDS_WITH' { + return recordFieldValue.endsWith(comparisonValue); + } + when else { + throw new IllegalArgumentException('Unsupported operator for String: ' + this.filterCondition.Operator__c); + } + } + } + } + + // Credit goes to this StackExchange post for the original BooleanExpression class - + // below is a modified version of the class + // https://salesforce.stackexchange.com/questions/113300/boolean-evaluation-in-apex/113308 + @SuppressWarnings('PMD.ApexDoc') + private class BooleanExpression { + public Boolean evaluate(String x) { + x = simplify(x); + + if (isSimpleExpression(x)) { + return Boolean.valueOf(x); + } + + if (x.contains('&&')) { + return andJoin(x.split('&&', 2)[0], x.split('&&', 2)[1]); + } + + if (x.contains('||')) { + String p1 = x.split('\\|\\|', 2)[0]; + String p2 = x.split('\\|\\|', 2)[1]; + + return orJoin(p1, p2); + } + + if (x.startsWith('!')) { + return !evaluate(x.substring(1)); + } + + return Boolean.valueOf(x); + } + + private Boolean orJoin(String x, String y) { + return evaluate(x) || evaluate(y); + } + + private Boolean andJoin(String x, String y) { + return evaluate(x) && evaluate(y); + } + + private Boolean isSimpleExpression(String x) { + return x == 'true' || x == 'false'; + } + + private String simplify(String x) { + x = x.trim(); + x = x.replace('AND', '&&'); + x = x.replace('OR', '||'); + while (x.contains('(') == true) { + String sub = x.substringAfterLast('(').substringBefore(')'); + x = x.replace('(' + sub + ')', String.valueOf(evaluate(sub))); + } + return x; + } + } + + @SuppressWarnings('PMD.ApexDoc') + private class FieldPath { + private List fieldChain; + private Schema.DescribeFieldResult fieldDescribe; + private String fieldPath; + private Schema.SObjectType sobjectType; + + public FieldPath(Schema.SObjectType sobjectType, String fieldPath) { + this.fieldChain = this.getFieldChain(sobjectType, fieldPath); + this.fieldPath = fieldPath; + + this.fieldDescribe = this.getLastFieldDescribe(); + } + + public Schema.SObjectField getField() { + return this.fieldChain[this.fieldChain.size() - 1]; + } + + public Object getValue(SObject record) { + Schema.SObjectType parentSObjectType = this.sobjectType; + SObject parentRecord = record; + + for (Schema.SObjectField field : this.fieldChain) { + Schema.DescribeFieldResult fieldDescribe = field.getDescribe(); + if (fieldDescribe.getSoapType() != Schema.SoapType.Id) { + return parentRecord.get(fieldDescribe.getName()); + } else { + parentSObjectType = fieldDescribe.getReferenceTo().get(0); + + SObject newParentRecord = parentRecord.getSObject(field); + if (newParentRecord == null) { + return null; + } else { + parentRecord = newParentRecord; + } + } + } + + return null; + } + + private List getFieldChain(Schema.SObjectType sobjectType, String fieldPath) { + Schema.SObjectType currentSObjectType = sobjectType; + + List fields = new List(); + List fieldPathPieces = fieldPath.split('\\.'); + Integer lastFieldIndex = fieldPathPieces.size() <= 1 ? 0 : fieldPathPieces.size() - 1; + + for (Integer i = 0; i < fieldPathPieces.size(); i++) { + String fieldPathPiece = fieldPathPieces[i]; + + String fieldApiName; + if (i == lastFieldIndex) { + fieldApiName = fieldPathPiece; + } else if (fieldPathPiece.endsWith('__r')) { + fieldApiName = fieldPathPiece.replace('__r', '__c'); + } else { + fieldApiName = fieldPathPiece + 'Id'; + } + + Schema.SObjectField field = currentSObjectType.getDescribe().fields.getMap().get(fieldApiName); + + // TODO add support for polymorphic fields + if (i < lastFieldIndex) { + currentSObjectType = field.getDescribe().getReferenceTo().get(0); + } + + fields.add(field); + } + + return fields; + } + + private Schema.DescribeFieldResult getLastFieldDescribe() { + Integer lastFieldIndex = this.fieldChain.size() - 1; + return this.fieldChain[lastFieldIndex].getDescribe(); + } + } +} diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls new file mode 100644 index 000000000..5f22a9bf3 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls @@ -0,0 +1,476 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.CyclomaticComplexity, PMD.ExcessiveParameterList, PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LogRetentionFilter_Tests { + @IsTest + static void it_should_match_one_condition_on_datetime_equal_to() { + Integer numberOfDaysToRetainLogs = 90; + Datetime mockTimestamp = System.now(); + LogRetentionRule__mdt rule = createMockRule('number_equal_to'); + rule.NumberOfDaysToRetainLogs__c = numberOfDaysToRetainLogs; + List conditions = new List{ + createMockRuleCondition('Timestamp__c', 'EQUAL_TO', 'Value', mockTimestamp) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.Timestamp__c = mockTimestamp; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(-9); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_datetime_not_equal_to() { + Integer numberOfDaysToRetainLogs = 90; + Datetime mockTimestamp = System.now(); + LogRetentionRule__mdt rule = createMockRule('number_not_equal_to'); + rule.NumberOfDaysToRetainLogs__c = numberOfDaysToRetainLogs; + List conditions = new List{ + createMockRuleCondition('Timestamp__c', 'NOT_EQUAL_TO', 'Value', mockTimestamp) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(-9); + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.Timestamp__c = mockTimestamp; + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_datetime_less_than() { + Integer numberOfDaysToRetainLogs = 90; + Datetime mockTimestamp = System.now(); + LogRetentionRule__mdt rule = createMockRule('number_less_than'); + rule.NumberOfDaysToRetainLogs__c = numberOfDaysToRetainLogs; + List conditions = new List{ + createMockRuleCondition('Timestamp__c', 'LESS_THAN', 'Value', mockTimestamp) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(-1); + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(+9); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_datetime_less_than_or_equal_to() { + Datetime mockTimestamp = System.now(); + LogRetentionRule__mdt rule = createMockRule('number_less_than_or_equal_to'); + List conditions = new List{ + createMockRuleCondition('Timestamp__c', 'LESS_THAN_OR_EQUAL_TO', 'Value', mockTimestamp) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.Timestamp__c = mockTimestamp; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(+9); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_datetime_greater_than() { + Datetime mockTimestamp = System.now(); + LogRetentionRule__mdt rule = createMockRule('number_greater_than'); + List conditions = new List{ + createMockRuleCondition('Timestamp__c', 'GREATER_THAN', 'Value', mockTimestamp) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(+9); + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(-1); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_datetime_greater_than_or_equal_to() { + Datetime mockTimestamp = System.now(); + LogRetentionRule__mdt rule = createMockRule('number_greater_than_or_equal_to'); + List conditions = new List{ + createMockRuleCondition('Timestamp__c', 'GREATER_THAN_OR_EQUAL_TO', 'Value', mockTimestamp) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.Timestamp__c = mockTimestamp; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.Timestamp__c = mockTimestamp.addMinutes(-9); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_throw_illegal_argument_exception_on_datetime_when_invalid_operator() { + String invalidOperator = 'THIS_IS_AN_INVALID_OPERATOR'; + Datetime mockTimestamp = System.now(); + LogRetentionRule__mdt rule = createMockRule('number_with_invalid_operator'); + List conditions = new List{ + createMockRuleCondition('Timestamp__c', invalidOperator, 'Value', mockTimestamp) + }; + LogEntry__c mockLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType).populateRequiredFields().getRecord(); + mockLogEntry.Timestamp__c = mockTimestamp; + + Exception thrownIllegalArgumentException; + try { + new LogRetentionFilter(mockLogEntry, rule, conditions).getFilterResult(); + } catch (IllegalArgumentException ex) { + thrownIllegalArgumentException = ex; + } + + System.assertNotEquals(null, thrownIllegalArgumentException); + System.assertEquals('Unsupported operator for Datetime: ' + invalidOperator, thrownIllegalArgumentException.getMessage()); + } + + @IsTest + static void it_should_match_one_condition_on_decimal_equal_to() { + Integer exampleLoggingLevelOrdinal = LoggingLevel.INFO.ordinal(); + LogRetentionRule__mdt rule = createMockRule('number_equal_to'); + List conditions = new List{ + createMockRuleCondition('LoggingLevelOrdinal__c', 'EQUAL_TO', 'Value', exampleLoggingLevelOrdinal) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal - 9; + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_decimal_not_equal_to() { + Integer exampleLoggingLevelOrdinal = LoggingLevel.INFO.ordinal(); + LogRetentionRule__mdt rule = createMockRule('number_not_equal_to'); + List conditions = new List{ + createMockRuleCondition('LoggingLevelOrdinal__c', 'NOT_EQUAL_TO', 'Value', exampleLoggingLevelOrdinal) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal - 9; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal; + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_decimal_less_than() { + Integer exampleLoggingLevelOrdinal = LoggingLevel.INFO.ordinal(); + LogRetentionRule__mdt rule = createMockRule('number_less_than'); + List conditions = new List{ + createMockRuleCondition('LoggingLevelOrdinal__c', 'LESS_THAN', 'Value', exampleLoggingLevelOrdinal) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal - 1; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal + 9; + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_decimal_less_than_or_equal_to() { + Integer exampleLoggingLevelOrdinal = LoggingLevel.INFO.ordinal(); + LogRetentionRule__mdt rule = createMockRule('number_less_than_or_equal_to'); + List conditions = new List{ + createMockRuleCondition('LoggingLevelOrdinal__c', 'LESS_THAN_OR_EQUAL_TO', 'Value', exampleLoggingLevelOrdinal) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal + 9; + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_decimal_greater_than() { + Integer exampleLoggingLevelOrdinal = LoggingLevel.INFO.ordinal(); + LogRetentionRule__mdt rule = createMockRule('number_greater_than'); + List conditions = new List{ + createMockRuleCondition('LoggingLevelOrdinal__c', 'GREATER_THAN', 'Value', exampleLoggingLevelOrdinal) + }; + LogEntry__c matchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + matchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal + 9; + LogEntry__c nonMatchingLogEntry = (LogEntry__c) LoggerMockDataCreator.createDataBuilder(Schema.LogEntry__c.SObjectType) + .populateRequiredFields() + .getRecord(); + nonMatchingLogEntry.LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal - 1; + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_decimal_greater_than_or_equal_to() { + Integer exampleLoggingLevelOrdinal = LoggingLevel.INFO.ordinal(); + LogRetentionRule__mdt rule = createMockRule('number_greater_than_or_equal_to'); + List conditions = new List{ + createMockRuleCondition('LoggingLevelOrdinal__c', 'GREATER_THAN_OR_EQUAL_TO', 'Value', exampleLoggingLevelOrdinal) + }; + LogEntry__c matchingLogEntry = new LogEntry__c(LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal); + LogEntry__c nonMatchingLogEntry = new LogEntry__c(LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal - 9); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_throw_illegal_argument_exception_on_decimal_when_invalid_operator() { + String invalidOperator = 'THIS_IS_AN_INVALID_OPERATOR'; + Integer exampleLoggingLevelOrdinal = LoggingLevel.INFO.ordinal(); + LogRetentionRule__mdt rule = createMockRule('number_with_invalid_operator'); + List conditions = new List{ + createMockRuleCondition('LoggingLevelOrdinal__c', invalidOperator, 'Value', exampleLoggingLevelOrdinal) + }; + LogEntry__c mockLogEntry = new LogEntry__c(LoggingLevelOrdinal__c = exampleLoggingLevelOrdinal); + + Exception thrownIllegalArgumentException; + try { + new LogRetentionFilter(mockLogEntry, rule, conditions).getFilterResult(); + } catch (IllegalArgumentException ex) { + thrownIllegalArgumentException = ex; + } + + System.assertNotEquals(null, thrownIllegalArgumentException); + System.assertEquals('Unsupported operator for Decimal: ' + invalidOperator, thrownIllegalArgumentException.getMessage()); + } + + @IsTest + static void it_should_match_one_condition_on_string_equal_to() { + String mockStringValue = 'Some text'; + LogRetentionRule__mdt rule = createMockRule('string_equal_to'); + List conditions = new List{ + createMockRuleCondition('Message__c', 'EQUAL_TO', 'Value', mockStringValue) + }; + LogEntry__c matchingLogEntry = new LogEntry__c(Message__c = mockStringValue); + LogEntry__c nonMatchingLogEntry = new LogEntry__c(Message__c = mockStringValue + ' some extra text so it does not match'); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_string_not_equal_to() { + String mockStringValue = 'Some text'; + LogRetentionRule__mdt rule = createMockRule('string_not_equal_to'); + List conditions = new List{ + createMockRuleCondition('Message__c', 'NOT_EQUAL_TO', 'Value', mockStringValue) + }; + LogEntry__c matchingLogEntry = new LogEntry__c(Message__c = mockStringValue + ' some extra text so it does not match'); + LogEntry__c nonMatchingLogEntry = new LogEntry__c(Message__c = mockStringValue); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_string_starts_with() { + String mockStringValue = 'Some text'; + LogRetentionRule__mdt rule = createMockRule('string_starts_with'); + List conditions = new List{ + createMockRuleCondition('Message__c', 'STARTS_WITH', 'Value', mockStringValue) + }; + LogEntry__c matchingLogEntry = new LogEntry__c(Message__c = mockStringValue + ' some extra text, but this should still match using STARTS_WITH'); + LogEntry__c nonMatchingLogEntry = new LogEntry__c(Message__c = 'This starts with something that does not match! ' + mockStringValue); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_string_contains() { + String mockStringValue = 'Some text'; + LogRetentionRule__mdt rule = createMockRule('string_contains'); + List conditions = new List{ + createMockRuleCondition('Message__c', 'CONTAINS', 'Value', mockStringValue) + }; + LogEntry__c matchingLogEntry = new LogEntry__c( + Message__c = 'Some extra text at the beginning ' + mockStringValue + ' some extra text, but this should still match using STARTS_WITH' + ); + LogEntry__c nonMatchingLogEntry = new LogEntry__c(Message__c = 'This does not contain the matching text at all!'); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_match_one_condition_on_string_ends_with() { + String mockStringValue = 'Some text'; + LogRetentionRule__mdt rule = createMockRule('string_ends_with', null); + List conditions = new List{ + createMockRuleCondition('Message__c', 'ENDS_WITH', 'Value', mockStringValue) + }; + LogEntry__c matchingLogEntry = new LogEntry__c(Message__c = 'Some extra text, but this should still match using ENDS_WITH' + mockStringValue); + LogEntry__c nonMatchingLogEntry = new LogEntry__c(Message__c = mockStringValue + ' and now this ends with something that does not match!'); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + @IsTest + static void it_should_throw_illegal_argument_exception_on_string_when_invalid_operator() { + String invalidOperator = 'THIS_IS_AN_INVALID_OPERATOR'; + String mockStringValue = 'Some text'; + LogRetentionRule__mdt rule = createMockRule('string_with_invalid_operator'); + List conditions = new List{ + createMockRuleCondition('Message__c', invalidOperator, 'Value', mockStringValue) + }; + LogEntry__c mockLogEntry = new LogEntry__c(Message__c = mockStringValue); + + Exception thrownIllegalArgumentException; + try { + new LogRetentionFilter(mockLogEntry, rule, conditions).getFilterResult(); + } catch (IllegalArgumentException ex) { + thrownIllegalArgumentException = ex; + } + + System.assertNotEquals(null, thrownIllegalArgumentException); + System.assertEquals('Unsupported operator for String: ' + invalidOperator, thrownIllegalArgumentException.getMessage()); + } + + @IsTest + static void it_should_match_one_condition_for_parent_field() { + String mockProfileName = 'Some profile'; + LogRetentionRule__mdt rule = createMockRule('string_equal_to'); + List conditions = new List{ + createMockRuleCondition('Log__r.ProfileName__c', 'EQUAL_TO', 'Value', mockProfileName) + }; + LogEntry__c matchingLogEntry = new LogEntry__c(Log__r = new Log__c(ProfileName__c = mockProfileName)); + LogEntry__c nonMatchingLogEntry = new LogEntry__c(Log__r = new Log__c(ProfileName__c = 'Some other value, that should not match')); + + LogRetentionFilter.FilterResult expectedMatchResult = new LogRetentionFilter(matchingLogEntry, rule, conditions).getFilterResult(); + LogRetentionFilter.FilterResult expectedNonMatchResult = new LogRetentionFilter(nonMatchingLogEntry, rule, conditions).getFilterResult(); + + System.assertEquals(true, expectedMatchResult.matchesFilter()); + System.assertEquals(false, expectedNonMatchResult.matchesFilter()); + } + + static LogRetentionRule__mdt createMockRule(String developerName) { + return createMockRule(developerName, null); + } + + static LogRetentionRule__mdt createMockRule(String developerName, String conditionLogicType) { + return new LogRetentionRule__mdt( + ConditionLogicType__c = conditionLogicType, + CustomConditionLogic__c = null, + DeveloperName = developerName, + IsEnabled__c = true + ); + } + + static LogRetentionRuleCondition__mdt createMockRuleCondition(String fieldPath, String operator, String valueType, Object value) { + String valueString = value instanceof String ? (String) value : JSON.serialize(value); + return new LogRetentionRuleCondition__mdt( + FieldPath__c = fieldPath, + Operator__c = operator, + SortOrder__c = null, + Value__c = valueString, + ValueType__c = valueType + ); + } +} diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionFilter_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls new file mode 100644 index 000000000..e6181852b --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls @@ -0,0 +1,130 @@ +//-----------------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// The core functionality of this plugin's code originated in https://github.com/jongpie/ApexValidationRules // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//-----------------------------------------------------------------------------------------------------------// + +/** + * @group Plugins + * @description Optional plugin that adds the ability to create & deploy advanced, configurable rules + * for setting the retention date of `Log__c` records, using custom metadata types + * `LogRetentionRule__mdt` and `LogRetentionRuleCondition__mdt`. + * @see LogRetentionFilter + */ +public without sharing class LogRetentionRulesPlugin implements LoggerPlugin.Triggerable { + private static final Map RULE_NAME_TO_RULE = new Map(); + private static final Map> RULE_NAME_TO_CONDITIONS = new Map>(); + + /** + * @description Handles converting Logger's buffer of `LogEntryEvent__e` records into `LogEntryArchive__b` records + * for any user with the included custom save method 'BIG_OBJECT' + * @param configuration The instance of `LoggerPlugin__mdt` configured for this specific plugin + * @param input The instance of `LoggerTriggerableContext`, provided by the logging system + */ + public void execute(LoggerPlugin__mdt configuration, LoggerTriggerableContext input) { + if (input.sobjectType != Schema.LogEntry__c.SObjectType) { + return; + } + + this.loadConfiguredFilters(); + + switch on input.triggerOperationType { + when AFTER_INSERT { + List logEntries = requeryLogEntries((List) input.triggerNew); + this.setLogRetentionDate(logEntries); + } + } + } + + private List requeryLogEntries(List logEntries) { + // Requery the LogEntry__c records so the parent Log__c fields can be included/used in rules + List logEntryFieldNames = new List(Schema.LogEntry__c.SObjectType.getDescribe().fields.getMap().keySet()); + + List logFieldNames = new List(Schema.Log__c.SObjectType.getDescribe().fields.getMap().keySet()); + logFieldNames.addAll(new List{ 'Owner.Name', 'Owner.Type' }); + for (String logFieldName : logFieldNames) { + String logRelationshipPrefix = Schema.LogEntry__c.Log__c.getDescribe().getName().substringBeforeLast('__c') + '__r'; + logEntryFieldNames.add(logRelationshipPrefix + '.' + logFieldName); + } + + String queryTemplate = 'SELECT {0} FROM {1} WHERE {2} IN :logEntries'; + List queryInputs = new List{ + String.join(logEntryFieldNames, ','), + Schema.LogEntry__c.SObjectType.getDescribe().getName(), + Schema.LogEntry__c.Id.getDescribe().getName() + }; + String logEntryQuery = String.escapeSingleQuotes(String.format(queryTemplate, queryInputs)); + return (List) Database.query(logEntryQuery); + } + + private void setLogRetentionDate(List logEntries) { + Map logIdToLog = new Map(); + List filterResults = this.runFilters(logEntries); + for (LogRetentionFilter.FilterResult filterResult : filterResults) { + if (filterResult.matchesFilter() == true) { + Id logId = (Id) filterResult.getRecord().get(Schema.LogEntry__c.Log__c); + Log__c log = logIdToLog.get(logId); + if (log == null) { + log = new Log__c(Id = logId); + } + + Integer numberOfDaysToRetainLogs = Integer.valueOf(filterResult.getRule().NumberOfDaysToRetainLogs__c); + log.LogRetentionDate__c = numberOfDaysToRetainLogs == null ? null : System.today().addDays(numberOfDaysToRetainLogs); + + logIdToLog.put(log.Id, log); + } + } + update logIdToLog.values(); + } + + @SuppressWarnings('PMD.ApexCRUDViolation') + private void loadConfiguredFilters() { + Map queriedRulesByDeveloperName = new Map(); + Map> queriedConditionsByRuleDeveloperName = new Map>(); + for (LogRetentionRule__mdt rule : [ + SELECT + DeveloperName, + ConditionLogicType__c, + CustomConditionLogic__c, + NumberOfDaysToRetainLogs__c, + (SELECT FieldPath__c, Operator__c, ValueType__c, Value__c FROM LogRetentionRuleConditions__r ORDER BY SortOrder__c NULLS LAST, DeveloperName) + FROM LogRetentionRule__mdt + WHERE IsEnabled__c = TRUE + ORDER BY ExecutionOrder__c NULLS LAST, DeveloperName + ]) { + queriedRulesByDeveloperName.put(rule.DeveloperName, rule); + queriedConditionsByRuleDeveloperName.put(rule.DeveloperName, rule.LogRetentionRuleConditions__r); + + if (System.Test.isRunningTest() == true) { + queriedRulesByDeveloperName.clear(); + queriedConditionsByRuleDeveloperName.clear(); + } + RULE_NAME_TO_RULE.putAll(queriedRulesByDeveloperName); + RULE_NAME_TO_CONDITIONS.putAll(queriedConditionsByRuleDeveloperName); + } + } + + private List runFilters(List records) { + List results = new List(); + for (SObject record : records) { + for (String filterDeveloperName : RULE_NAME_TO_RULE.keySet()) { + LogRetentionRule__mdt filter = RULE_NAME_TO_RULE.get(filterDeveloperName); + List filterConditions = RULE_NAME_TO_CONDITIONS.get(filter.DeveloperName); + LogRetentionFilter.FilterResult filterResult = new LogRetentionFilter(record, filter, filterConditions).getFilterResult(); + + results.add(filterResult); + } + } + return results; + } + + @TestVisible + private static void setMockRetentionRule(LogRetentionRule__mdt rule) { + RULE_NAME_TO_RULE.put(rule.DeveloperName, rule); + } + + @TestVisible + private static void setMockRetentionRuleConditions(LogRetentionRule__mdt rule, List conditions) { + RULE_NAME_TO_CONDITIONS.put(rule.DeveloperName, conditions); + } +} diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls new file mode 100644 index 000000000..d40c190ad --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls @@ -0,0 +1,211 @@ +//-----------------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// The core functionality of this plugin's code originated in https://github.com/jongpie/ApexValidationRules // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//-----------------------------------------------------------------------------------------------------------// + +// TODO revise suppressed PMD rules/clean up code +@SuppressWarnings('PMD.ApexDoc, PMD.ExcessiveParameterList, PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class LogRetentionRulesPlugin_Tests { + @IsTest + static void it_should_set_retention_date_for_rule_with_one_condition_when_datetime_less_than() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + enablePlugin(); + Date originalLogRetentionDate = System.today().addDays(Integer.valueOf(Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c)); + Integer numberOfDaysToRetainLogs = 90; + Date expectedLogRetentionDate = System.today().addDays(numberOfDaysToRetainLogs); + Datetime now = System.now(); + LogRetentionRule__mdt rule = createMockRule('rule_with_multiple_AND_conditions', 'AND', numberOfDaysToRetainLogs); + rule.NumberOfDaysToRetainLogs__c = numberOfDaysToRetainLogs; + LogRetentionRulesPlugin.setMockRetentionRule(rule); + List conditions = new List{ + createMockRuleCondition('Log__r.CreatedDate', 'LESS_THAN', 'Value', now) + }; + LogRetentionRulesPlugin.setMockRetentionRuleConditions(rule, conditions); + Log__c log = new Log__c(TransactionId__c = '1234'); + insert log; + Test.setCreatedDate(log.Id, now.addDays(-1)); + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalLogRetentionDate, log.LogRetentionDate__c); + + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.ERROR.name(), TransactionEntryNumber__c = 1); + insert logEntry; + + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(expectedLogRetentionDate, log.LogRetentionDate__c); + } + + @IsTest + static void it_should_set_retention_date_for_rule_with_one_condition_when_datetime_less_than_or_equal_to() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + enablePlugin(); + Date originalLogRetentionDate = System.today().addDays(Integer.valueOf(Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c)); + Integer numberOfDaysToRetainLogs = 90; + Date expectedLogRetentionDate = System.today().addDays(numberOfDaysToRetainLogs); + Datetime now = System.now(); + LogRetentionRule__mdt rule = createMockRule('rule_with_multiple_AND_conditions', 'AND', numberOfDaysToRetainLogs); + rule.NumberOfDaysToRetainLogs__c = numberOfDaysToRetainLogs; + LogRetentionRulesPlugin.setMockRetentionRule(rule); + List conditions = new List{ + createMockRuleCondition('Log__r.CreatedDate', 'LESS_THAN_OR_EQUAL_TO', 'Value', now) + }; + LogRetentionRulesPlugin.setMockRetentionRuleConditions(rule, conditions); + + Log__c log = new Log__c(TransactionId__c = '1234'); + insert log; + Test.setCreatedDate(log.Id, now); + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalLogRetentionDate, log.LogRetentionDate__c); + + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.ERROR.name(), TransactionEntryNumber__c = 1); + insert logEntry; + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(expectedLogRetentionDate, log.LogRetentionDate__c); + } + + @IsTest + static void it_should_set_retention_date_for_rule_with_one_string_equality_condition() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + enablePlugin(); + Date originalLogRetentionDate = System.today().addDays(Integer.valueOf(Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c)); + Integer numberOfDaysToRetainLogs = 90; + Date expectedLogRetentionDate = System.today().addDays(numberOfDaysToRetainLogs); + String scenario = 'Some scenario'; + LogRetentionRule__mdt rule = createMockRule('rule_with_multiple_AND_conditions', 'AND', numberOfDaysToRetainLogs); + rule.NumberOfDaysToRetainLogs__c = numberOfDaysToRetainLogs; + LogRetentionRulesPlugin.setMockRetentionRule(rule); + List conditions = new List{ + createMockRuleCondition('Log__r.Scenario__c', 'EQUAL_TO', 'Value', scenario) + }; + LogRetentionRulesPlugin.setMockRetentionRuleConditions(rule, conditions); + + Log__c log = new Log__c(Scenario__c = scenario, TransactionId__c = '1234'); + insert log; + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalLogRetentionDate, log.LogRetentionDate__c); + + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.ERROR.name(), TransactionEntryNumber__c = 1); + insert logEntry; + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(expectedLogRetentionDate, log.LogRetentionDate__c); + } + + @IsTest + static void it_should_set_retention_date_for_rule_with_multiple_and_conditions() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + enablePlugin(); + Date originalLogRetentionDate = System.today().addDays(Integer.valueOf(Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c)); + Integer numberOfDaysToRetainLogs = 90; + Date expectedLogRetentionDate = System.today().addDays(numberOfDaysToRetainLogs); + String scenario = 'Some scenario'; + Integer userLoggingLevelOrdinal = LoggingLevel.WARN.ordinal(); + LogRetentionRule__mdt rule = createMockRule('rule_with_multiple_AND_conditions', 'AND', numberOfDaysToRetainLogs); + LogRetentionRulesPlugin.setMockRetentionRule(rule); + List conditions = new List{ + createMockRuleCondition('Log__r.Scenario__c', 'EQUAL_TO', 'Value', scenario), + createMockRuleCondition('Log__r.UserLoggingLevelOrdinal__c', 'GREATER_THAN_OR_EQUAL_TO', 'Value', userLoggingLevelOrdinal) + }; + LogRetentionRulesPlugin.setMockRetentionRuleConditions(rule, conditions); + + Log__c log = new Log__c(Scenario__c = scenario, TransactionId__c = '1234', UserLoggingLevelOrdinal__c = userLoggingLevelOrdinal); + insert log; + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalLogRetentionDate, log.LogRetentionDate__c); + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.ERROR.name(), TransactionEntryNumber__c = 1); + insert logEntry; + + log = [SELECT Id, LogRetentionDate__c, Scenario__c, TotalERRORLogEntries__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(expectedLogRetentionDate, log.LogRetentionDate__c, log); + } + + @IsTest + static void it_should_set_retention_date_for_rule_with_multiple_or_conditions() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + enablePlugin(); + Date originalLogRetentionDate = System.today().addDays(Integer.valueOf(Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c)); + Integer numberOfDaysToRetainLogs = 90; + Date expectedLogRetentionDate = System.today().addDays(numberOfDaysToRetainLogs); + String scenario1 = 'Some scenario'; + String scenario2 = 'Another scenario'; + LogRetentionRule__mdt rule = createMockRule('rule_with_multiple_OR_conditions', 'OR', numberOfDaysToRetainLogs); + LogRetentionRulesPlugin.setMockRetentionRule(rule); + List conditions = new List{ + createMockRuleCondition('Log__r.Scenario__c', 'EQUAL_TO', 'Value', scenario1), + createMockRuleCondition('Log__r.Scenario__c', 'EQUAL_TO', 'Value', scenario2) + }; + LogRetentionRulesPlugin.setMockRetentionRuleConditions(rule, conditions); + + Log__c log = new Log__c(Scenario__c = scenario1, TransactionId__c = '1234'); + insert log; + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalLogRetentionDate, log.LogRetentionDate__c); + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.ERROR.name(), TransactionEntryNumber__c = 1); + insert logEntry; + + log = [SELECT Id, LogRetentionDate__c, Scenario__c, TotalERRORLogEntries__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(expectedLogRetentionDate, log.LogRetentionDate__c, log); + } + + @IsTest + static void it_should_set_retention_date_for_rule_with_multiple_custom_conditions() { + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + enablePlugin(); + Date originalLogRetentionDate = System.today().addDays(Integer.valueOf(Logger.getUserSettings().DefaultNumberOfDaysToRetainLogs__c)); + Integer numberOfDaysToRetainLogs = 90; + Date expectedLogRetentionDate = System.today().addDays(numberOfDaysToRetainLogs); + String scenario1 = 'Some scenario'; + String scenario2 = 'Another scenario'; + Integer userLoggingLevelOrdinal = LoggingLevel.WARN.ordinal(); + LogRetentionRule__mdt rule = createMockRule('rule_with_multiple_OR_conditions', 'Custom', numberOfDaysToRetainLogs); + rule.CustomConditionLogic__c = '((1 OR 2) AND 3)'; + LogRetentionRulesPlugin.setMockRetentionRule(rule); + List conditions = new List{ + createMockRuleCondition('Log__r.Scenario__c', 'EQUAL_TO', 'Value', scenario1), + createMockRuleCondition('Log__r.Scenario__c', 'EQUAL_TO', 'Value', scenario2), + createMockRuleCondition('Log__r.UserLoggingLevelOrdinal__c', 'GREATER_THAN_OR_EQUAL_TO', 'Value', userLoggingLevelOrdinal) + }; + LogRetentionRulesPlugin.setMockRetentionRuleConditions(rule, conditions); + + Log__c log = new Log__c(Scenario__c = scenario1, TransactionId__c = '1234', UserLoggingLevelOrdinal__c = userLoggingLevelOrdinal); + insert log; + log = [SELECT Id, LogRetentionDate__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(originalLogRetentionDate, log.LogRetentionDate__c); + LogEntry__c logEntry = new LogEntry__c(Log__c = log.Id, LoggingLevel__c = LoggingLevel.ERROR.name(), TransactionEntryNumber__c = 1); + insert logEntry; + + log = [SELECT Id, LogRetentionDate__c, Scenario__c, TotalERRORLogEntries__c FROM Log__c WHERE Id = :log.Id]; + System.assertEquals(expectedLogRetentionDate, log.LogRetentionDate__c, log); + } + + static void enablePlugin() { + // Set the plugin's parameters + LoggerPlugin__mdt pluginConfiguration = new LoggerPlugin__mdt( + DeveloperName = 'LogRetentionRulesPlugin', + IsEnabled__c = true, + SObjectHandlerApexClass__c = LogRetentionRulesPlugin.class.getName() + ); + LoggerTestConfigurator.setMock(pluginConfiguration); + } + + static LogRetentionRule__mdt createMockRule(String developerName, String conditionLogicType, Integer numberOfDaysToRetainLogs) { + return new LogRetentionRule__mdt( + ConditionLogicType__c = conditionLogicType, + CustomConditionLogic__c = null, + DeveloperName = developerName, + IsEnabled__c = true, + NumberOfDaysToRetainLogs__c = numberOfDaysToRetainLogs + ); + } + + static LogRetentionRuleCondition__mdt createMockRuleCondition(String fieldPath, String operator, String valueType, Object value) { + String valueString = value instanceof String ? (String) value : JSON.serialize(value); + return new LogRetentionRuleCondition__mdt( + FieldPath__c = fieldPath, + Operator__c = operator, + SortOrder__c = null, + Value__c = valueString, + ValueType__c = valueType + ); + } +} diff --git a/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/classes/LogRetentionRulesPlugin_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Error_Logs.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Error_Logs.md-meta.xml new file mode 100644 index 000000000..ea8af3388 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Error_Logs.md-meta.xml @@ -0,0 +1,29 @@ + + + + false + + ConditionLogicType__c + AND + + + CustomConditionLogic__c + + + + ExecutionOrder__c + + + + IsEnabled__c + true + + + NumberOfDaysToRetainLogs__c + 60.0 + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Scenarios_With_Errors.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Scenarios_With_Errors.md-meta.xml new file mode 100644 index 000000000..6cab5ffdf --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_1_Scenarios_With_Errors.md-meta.xml @@ -0,0 +1,29 @@ + + + + false + + ConditionLogicType__c + Custom + + + CustomConditionLogic__c + 1 AND (2 OR 3) + + + ExecutionOrder__c + 1.0 + + + IsEnabled__c + true + + + NumberOfDaysToRetainLogs__c + 90.0 + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_2_Additional_Error_Logs.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_2_Additional_Error_Logs.md-meta.xml new file mode 100644 index 000000000..dfadbb02c --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRule.Sample_Rule_2_Additional_Error_Logs.md-meta.xml @@ -0,0 +1,29 @@ + + + + false + + ConditionLogicType__c + AND + + + CustomConditionLogic__c + + + + ExecutionOrder__c + 2.0 + + + IsEnabled__c + true + + + NumberOfDaysToRetainLogs__c + 60.0 + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Error_Condition.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Error_Condition.md-meta.xml new file mode 100644 index 000000000..6d3257d3c --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Error_Condition.md-meta.xml @@ -0,0 +1,33 @@ + + + + false + + FieldPath__c + Log__r.TotalERRORLogEntries__c + + + LogRetentionRule__c + Sample_Rule_1_Scenarios_With_Errors + + + Operator__c + GREATER_THAN_OR_EQUAL_TO + + + SortOrder__c + 1.0 + + + ValueType__c + Value + + + Value__c + 1 + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_A_Condition.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_A_Condition.md-meta.xml new file mode 100644 index 000000000..3295e724e --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_A_Condition.md-meta.xml @@ -0,0 +1,33 @@ + + + + false + + FieldPath__c + Log__r.Scenario__c + + + LogRetentionRule__c + Sample_Rule_1_Scenarios_With_Errors + + + Operator__c + EQUAL_TO + + + SortOrder__c + 2.0 + + + ValueType__c + Value + + + Value__c + feature A + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_B_Condition.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_B_Condition.md-meta.xml new file mode 100644 index 000000000..8354ab432 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_Scenario_B_Condition.md-meta.xml @@ -0,0 +1,33 @@ + + + + false + + FieldPath__c + Log__r.Scenario__c + + + LogRetentionRule__c + Sample_Rule_1_Scenarios_With_Errors + + + Operator__c + EQUAL_TO + + + SortOrder__c + 3.0 + + + ValueType__c + Value + + + Value__c + feature B + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_condition.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_condition.md-meta.xml new file mode 100644 index 000000000..9ba8cdc0f --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_1_condition.md-meta.xml @@ -0,0 +1,33 @@ + + + + false + + FieldPath__c + Log__r.TotalERRORLogEntries__c + + + LogRetentionRule__c + Sample_Rule_1_Error_Logs + + + Operator__c + GREATER_THAN + + + SortOrder__c + + + + ValueType__c + Value + + + Value__c + 1 + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_2_Error_Condition.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_2_Error_Condition.md-meta.xml new file mode 100644 index 000000000..d40504c77 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LogRetentionRuleCondition.Sample_Rule_2_Error_Condition.md-meta.xml @@ -0,0 +1,33 @@ + + + + false + + FieldPath__c + Log__r.TotalERRORLogEntries__c + + + LogRetentionRule__c + Sample_Rule_2_Additional_Error_Logs + + + Operator__c + GREATER_THAN + + + SortOrder__c + + + + ValueType__c + Value + + + Value__c + 1 + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LoggerPlugin.LogRetentionRules.md-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LoggerPlugin.LogRetentionRules.md-meta.xml new file mode 100644 index 000000000..867b48be3 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/customMetadata/LoggerPlugin.LogRetentionRules.md-meta.xml @@ -0,0 +1,51 @@ + + + + false + + BatchPurgerApexClass__c + + + + BatchPurgerExecutionOrder__c + + + + BatchPurgerFlowName__c + + + + Description__c + Adds the ability to create & deploy advanced, configurable rules for setting the retention date of Log__c records, using custom metadata types LogRetentionRule__mdt and LogRetentionRuleCondition__mdt. + + + IsEnabled__c + true + + + Link__c + + + + SObjectHandlerApexClass__c + + + + SObjectHandlerExecutionOrder__c + + + + SObjectHandlerFlowName__c + + + + VersionNumber__c + v0.9.0 + + diff --git a/nebula-logger/core/main/plugin-framework/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/layouts/LogRetentionRuleCondition__mdt-Log Retention Rule Condition Layout.layout-meta.xml similarity index 82% rename from nebula-logger/core/main/plugin-framework/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml rename to nebula-logger/plugins/log-retention-rules/plugin/layouts/LogRetentionRuleCondition__mdt-Log Retention Rule Condition Layout.layout-meta.xml index 3f2bf15a1..e36a1256d 100644 --- a/nebula-logger/core/main/plugin-framework/layouts/LoggerPlugin__mdt-Logger Plugin Layout.layout-meta.xml +++ b/nebula-logger/plugins/log-retention-rules/plugin/layouts/LogRetentionRuleCondition__mdt-Log Retention Rule Condition Layout.layout-meta.xml @@ -16,9 +16,13 @@ + + Required + LogRetentionRule__c + Edit - IsEnabled__c + SortOrder__c @@ -27,41 +31,28 @@ true true true - + Required - SObjectType__c + FieldPath__c - Edit - ExecutionOrder__c + Required + Operator__c Required - PluginType__c - - - Required - PluginApiName__c + ValueType__c - - - - - true - false - false - - Edit - Description__c + Value__c - + false @@ -70,8 +61,8 @@ - Required - NamespacePrefix + Edit + IsProtected Readonly @@ -80,8 +71,8 @@ - Edit - IsProtected + Required + NamespacePrefix Readonly @@ -106,7 +97,7 @@ false false - 00h17000007ebex + 00h1F000005cUjP 4 0 Default diff --git a/nebula-logger/plugins/log-retention-rules/plugin/layouts/LogRetentionRule__mdt-Log Retention Rule Layout.layout-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/layouts/LogRetentionRule__mdt-Log Retention Rule Layout.layout-meta.xml new file mode 100644 index 000000000..958d04dad --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/layouts/LogRetentionRule__mdt-Log Retention Rule Layout.layout-meta.xml @@ -0,0 +1,116 @@ + + + + false + true + true + + + + Required + MasterLabel + + + Required + DeveloperName + + + + + Edit + IsEnabled__c + + + Edit + ExecutionOrder__c + + + + + + true + true + true + + + + Edit + NumberOfDaysToRetainLogs__c + + + + + Required + ConditionLogicType__c + + + Edit + CustomConditionLogic__c + + + + + + false + true + true + + + + Edit + IsProtected + + + Readonly + CreatedById + + + + + Edit + NamespacePrefix + + + Readonly + LastModifiedById + + + + + + true + true + false + + + + + + + + NumberOfDaysToRetainLogs__c + + + MasterLabel + DeveloperName + SortOrder__c + FieldPath__c + Operator__c + ValueType__c + Value__c + LogRetentionRuleCondition__mdt.LogRetentionRule__c + SortOrder__c + Asc + + false + false + false + false + false + + 00h17000007xeHS + 4 + 0 + Default + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/LogRetentionRuleCondition__mdt.object-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/LogRetentionRuleCondition__mdt.object-meta.xml new file mode 100644 index 000000000..24cdcfb5d --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/LogRetentionRuleCondition__mdt.object-meta.xml @@ -0,0 +1,8 @@ + + + Used to configure field-level conditions for retention rules - each condition checks a LogEntry__c or Log__c field for a specific value, regular expression (regex), or field comparisons. + + Log Retention Rule Conditions + Public + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/FieldPath__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/FieldPath__c.field-meta.xml new file mode 100644 index 000000000..4724e4a9e --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/FieldPath__c.field-meta.xml @@ -0,0 +1,12 @@ + + + FieldPath__c + false + SubscriberControlled + The API name of the LogEntryEvent__e field. Note: parent field, such as CreatedBy.Name, are not supported. + + 255 + true + Text + false + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/LogRetentionRule__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/LogRetentionRule__c.field-meta.xml new file mode 100644 index 000000000..d89974c73 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/LogRetentionRule__c.field-meta.xml @@ -0,0 +1,13 @@ + + + LogRetentionRule__c + false + SubscriberControlled + + LogRetentionRule__mdt + Log Retention Rule Conditions + LogRetentionRuleConditions + true + MetadataRelationship + false + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Operator__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Operator__c.field-meta.xml new file mode 100644 index 000000000..b62486bba --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Operator__c.field-meta.xml @@ -0,0 +1,60 @@ + + + Operator__c + false + SubscriberControlled + + true + Picklist + + true + + false + + EQUAL_TO + false + + + + NOT_EQUAL_TO + false + + + + LESS_THAN + false + + + + GREATER_THAN + false + + + + LESS_THAN_OR_EQUAL_TO + false + + + + GREATER_THAN_OR_EQUAL_TO + false + + + + STARTS_WITH + false + + + + CONTAINS + false + + + + ENDS_WITH + false + + + + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/SortOrder__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/SortOrder__c.field-meta.xml new file mode 100644 index 000000000..78d7b35b1 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/SortOrder__c.field-meta.xml @@ -0,0 +1,12 @@ + + + SortOrder__c + false + SubscriberControlled + + 3 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/ValueType__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/ValueType__c.field-meta.xml new file mode 100644 index 000000000..54ea1d6d1 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/ValueType__c.field-meta.xml @@ -0,0 +1,30 @@ + + + ValueType__c + false + SubscriberControlled + + true + Picklist + + true + + false + + Value + false + + + + Field + false + + + + RegEx + false + + + + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Value__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Value__c.field-meta.xml new file mode 100644 index 000000000..0e1738715 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/fields/Value__c.field-meta.xml @@ -0,0 +1,11 @@ + + + Value__c + false + SubscriberControlled + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/listViews/All.listView-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/listViews/All.listView-meta.xml new file mode 100644 index 000000000..0b411c06f --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRuleCondition__mdt/listViews/All.listView-meta.xml @@ -0,0 +1,14 @@ + + + All + MasterLabel + DeveloperName + LogRetentionRule__c + SortOrder__c + FieldPath__c + Operator__c + ValueType__c + Value__c + Everything + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/LogRetentionRule__mdt.object-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/LogRetentionRule__mdt.object-meta.xml new file mode 100644 index 000000000..95a15408d --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/LogRetentionRule__mdt.object-meta.xml @@ -0,0 +1,8 @@ + + + Used to configure rules that set the value of Log__c.LogRetentionDate__c. Each rules consists of 1 or more conditions, stored in LogRetentionRuleCondition__mdt. + + Log Retention Rules + Public + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ConditionLogicType__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ConditionLogicType__c.field-meta.xml new file mode 100644 index 000000000..e122b5298 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ConditionLogicType__c.field-meta.xml @@ -0,0 +1,30 @@ + + + ConditionLogicType__c + false + SubscriberControlled + + true + Picklist + + true + + false + + AND + false + + + + OR + false + + + + Custom + false + + + + + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/CustomConditionLogic__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/CustomConditionLogic__c.field-meta.xml new file mode 100644 index 000000000..cb30f8610 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/CustomConditionLogic__c.field-meta.xml @@ -0,0 +1,11 @@ + + + CustomConditionLogic__c + false + SubscriberControlled + + 255 + false + Text + false + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ExecutionOrder__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ExecutionOrder__c.field-meta.xml new file mode 100644 index 000000000..d37bda01f --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/ExecutionOrder__c.field-meta.xml @@ -0,0 +1,14 @@ + + + ExecutionOrder__c + false + SubscriberControlled + The specific order to execute the retention rules in the org. Rules are executed in order by sorting by execution order, then rule name (ORDER BY ExecutionOrder__c NULLS LAST, DeveloperName). In the event that multiple rules apply to a record, the first matching rule is used to set the log retention date. + + 3 + false + 0 + Number + false + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/IsEnabled__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/IsEnabled__c.field-meta.xml new file mode 100644 index 000000000..7c3dca299 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/IsEnabled__c.field-meta.xml @@ -0,0 +1,9 @@ + + + IsEnabled__c + true + false + SubscriberControlled + + Checkbox + diff --git a/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/NumberOfDaysToRetainLogs__c.field-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/NumberOfDaysToRetainLogs__c.field-meta.xml new file mode 100644 index 000000000..613a84788 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/fields/NumberOfDaysToRetainLogs__c.field-meta.xml @@ -0,0 +1,14 @@ + + + NumberOfDaysToRetainLogs__c + false + SubscriberControlled + This value is used to set the field Log__c.LogRetentionDate__c, which is then used by LogBatchPurger to delete old logs. To keep logs indefinitely, set this field to blank (null). + + 18 + false + 0 + Number + false + diff --git a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/listViews/All.listView-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/listViews/All.listView-meta.xml similarity index 69% rename from nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/listViews/All.listView-meta.xml rename to nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/listViews/All.listView-meta.xml index eaa65e95c..ab54559ea 100644 --- a/nebula-logger/core/main/plugin-framework/objects/LoggerPlugin__mdt/listViews/All.listView-meta.xml +++ b/nebula-logger/plugins/log-retention-rules/plugin/objects/LogRetentionRule__mdt/listViews/All.listView-meta.xml @@ -4,11 +4,10 @@ MasterLabel DeveloperName IsEnabled__c + ConditionLogicType__c + CustomConditionLogic__c + NumberOfDaysToRetainLogs__c ExecutionOrder__c - SObjectType__c - PluginType__c - PluginApiName__c - Description__c Everything diff --git a/nebula-logger/plugins/log-retention-rules/plugin/testSuites/LoggerLogRetentionRulesPlugin.testSuite-meta.xml b/nebula-logger/plugins/log-retention-rules/plugin/testSuites/LoggerLogRetentionRulesPlugin.testSuite-meta.xml new file mode 100644 index 000000000..aa8f4f1c9 --- /dev/null +++ b/nebula-logger/plugins/log-retention-rules/plugin/testSuites/LoggerLogRetentionRulesPlugin.testSuite-meta.xml @@ -0,0 +1,5 @@ + + + LogRetentionFilter_Tests + LogRetentionRulesPlugin_Tests + diff --git a/nebula-logger/plugins/Slack/images/btn-install-unlocked-package-plugin.png b/nebula-logger/plugins/logger-admin-dashboard/.images/btn-install-unlocked-package-plugin.png similarity index 100% rename from nebula-logger/plugins/Slack/images/btn-install-unlocked-package-plugin.png rename to nebula-logger/plugins/logger-admin-dashboard/.images/btn-install-unlocked-package-plugin.png diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/images/logger-admin-dashboard.png b/nebula-logger/plugins/logger-admin-dashboard/.images/logger-admin-dashboard.png similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/images/logger-admin-dashboard.png rename to nebula-logger/plugins/logger-admin-dashboard/.images/logger-admin-dashboard.png diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/README.md b/nebula-logger/plugins/logger-admin-dashboard/README.md similarity index 64% rename from nebula-logger/plugins/Logger-Admin-Dashboard/README.md rename to nebula-logger/plugins/logger-admin-dashboard/README.md index e5674cfbf..c463c7c88 100644 --- a/nebula-logger/plugins/Logger-Admin-Dashboard/README.md +++ b/nebula-logger/plugins/logger-admin-dashboard/README.md @@ -1,8 +1,8 @@ # Logger Admin Dashboard for Nebula Logger -> :information_source: Requires Nebula Logger v4.6.12 or newer +> :information_source: This plugin requires `v4.6.12` or newer of Nebula Logger's unlocked package -[![Install Unlocked Package Plugin](./images/btn-install-unlocked-package-plugin.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015l3yQAA) +[![Install Unlocked Package Plugin](../.images/btn-install-unlocked-package-plugin-sandbox.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015l3yQAA) To help monitor logs in your org, this plugin includes a new Logger Admin dynamic dashboard, as well as several new reports. The dashboard displays several report charts to quickly see summary info about your logs. It supports filtering on: diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/dashboards/LogDashboards.dashboardFolder-meta.xml b/nebula-logger/plugins/logger-admin-dashboard/plugin/dashboards/LogDashboards.dashboardFolder-meta.xml similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/plugin/dashboards/LogDashboards.dashboardFolder-meta.xml rename to nebula-logger/plugins/logger-admin-dashboard/plugin/dashboards/LogDashboards.dashboardFolder-meta.xml diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/dashboards/LogDashboards/LoggerAdmin.dashboard-meta.xml b/nebula-logger/plugins/logger-admin-dashboard/plugin/dashboards/LogDashboards/LoggerAdmin.dashboard-meta.xml similarity index 99% rename from nebula-logger/plugins/Logger-Admin-Dashboard/plugin/dashboards/LogDashboards/LoggerAdmin.dashboard-meta.xml rename to nebula-logger/plugins/logger-admin-dashboard/plugin/dashboards/LogDashboards/LoggerAdmin.dashboard-meta.xml index e0d4d6f2f..55ee66a79 100644 --- a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/dashboards/LogDashboards/LoggerAdmin.dashboard-meta.xml +++ b/nebula-logger/plugins/logger-admin-dashboard/plugin/dashboards/LogDashboards/LoggerAdmin.dashboard-meta.xml @@ -335,6 +335,7 @@ LoggedInUser true + test-ghsv1imtoehu@example.com #000000 Logger Admin Dashboard #000000 diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryDailyRetentionSummary.report-meta.xml b/nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryDailyRetentionSummary.report-meta.xml similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryDailyRetentionSummary.report-meta.xml rename to nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryDailyRetentionSummary.report-meta.xml diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryDailySummary.report-meta.xml b/nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryDailySummary.report-meta.xml similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryDailySummary.report-meta.xml rename to nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryDailySummary.report-meta.xml diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryOriginSummary.report-meta.xml b/nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryOriginSummary.report-meta.xml similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryOriginSummary.report-meta.xml rename to nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryOriginSummary.report-meta.xml diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryScenarioSummary.report-meta.xml b/nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryScenarioSummary.report-meta.xml similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntryScenarioSummary.report-meta.xml rename to nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntryScenarioSummary.report-meta.xml diff --git a/nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntrySummary.report-meta.xml b/nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntrySummary.report-meta.xml similarity index 100% rename from nebula-logger/plugins/Logger-Admin-Dashboard/plugin/reports/LogReports/LogEntrySummary.report-meta.xml rename to nebula-logger/plugins/logger-admin-dashboard/plugin/reports/LogReports/LogEntrySummary.report-meta.xml diff --git a/nebula-logger/plugins/Slack/images/slack-plugin-notification.png b/nebula-logger/plugins/slack/.images/slack-plugin-notification.png similarity index 100% rename from nebula-logger/plugins/Slack/images/slack-plugin-notification.png rename to nebula-logger/plugins/slack/.images/slack-plugin-notification.png diff --git a/nebula-logger/plugins/Slack/images/slack-plugin-parameters.png b/nebula-logger/plugins/slack/.images/slack-plugin-parameters.png similarity index 100% rename from nebula-logger/plugins/Slack/images/slack-plugin-parameters.png rename to nebula-logger/plugins/slack/.images/slack-plugin-parameters.png diff --git a/nebula-logger/plugins/Slack/README.md b/nebula-logger/plugins/slack/README.md similarity index 79% rename from nebula-logger/plugins/Slack/README.md rename to nebula-logger/plugins/slack/README.md index 7f5c5646c..07fdb637f 100644 --- a/nebula-logger/plugins/Slack/README.md +++ b/nebula-logger/plugins/slack/README.md @@ -1,8 +1,10 @@ # Slack plugin for Nebula Logger -Adds a Slack integration for the unlocked package edition of Nebula Logger v4.6.9 or newer. Any logs with log entries that meet a certain (configurable) logging level will automatically be posted to your Slack channel via an asynchronous `Queueable` job. +> :information_source: This plugin requires `v4.7.1` or newer of Nebula Logger's unlocked package -[![Install Unlocked Package](./images/btn-install-unlocked-package-plugin.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015l2WQAQ) +[![Install Unlocked Package](../.images/btn-install-unlocked-package-plugin-sandbox.png)](https://test.salesforce.com/packaging/installPackage.apexp?p0=04t5Y0000015lgQQAQ) + +Adds a Slack integration for the unlocked package edition of Nebula Logger. Any logs with log entries that meet a certain (configurable) logging level will automatically be posted to your Slack channel via an asynchronous `Queueable` job. ![Slack plugin: notification](./images/slack-plugin-notification.png) @@ -10,13 +12,13 @@ Adds a Slack integration for the unlocked package edition of Nebula Logger v4.6. ## What's Included -This plugin includes some add-on metadata for Logger to support the Slack integration +This plugin includes some add-on metadata for Nebula Logger to support the Slack integration 1. Apex class `SlackLoggerPlugin` and corresponding tests in `SlackLoggerPlugin_Tests` 2. Plugin configuration details stored in Logger's CMDT objects `LoggerPlugin__mdt` and `LoggerParameter__mdt` 3. Custom fields `Log__c.SendSlackNotification__c` and `Log__c.SlackNotificationDate__c` 4. Field-level security (FLS) via a new permission set `LoggerSlackPluginAdmin` to provide access to the custom Slack fields -5. Custom list views for the `Log__c` and `LoggerParameter__mdt` objects +5. Two custom list views for the `Log__c` object to see any `Log__c` records that have, or should be, sent to Slack 6. Remote site setting for Slack's API --- diff --git a/nebula-logger/plugins/Slack/plugin/core/main/README.md b/nebula-logger/plugins/slack/plugin/core/main/README.md similarity index 100% rename from nebula-logger/plugins/Slack/plugin/core/main/README.md rename to nebula-logger/plugins/slack/plugin/core/main/README.md diff --git a/nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/fields/SendSlackNotification__c.field-meta.xml b/nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/fields/SendSlackNotification__c.field-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/fields/SendSlackNotification__c.field-meta.xml rename to nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/fields/SendSlackNotification__c.field-meta.xml diff --git a/nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/fields/SlackNotificationDate__c.field-meta.xml b/nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/fields/SlackNotificationDate__c.field-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/fields/SlackNotificationDate__c.field-meta.xml rename to nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/fields/SlackNotificationDate__c.field-meta.xml diff --git a/nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsSentToSlack.listView-meta.xml b/nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsSentToSlack.listView-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsSentToSlack.listView-meta.xml rename to nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsSentToSlack.listView-meta.xml diff --git a/nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsToBeSentToSlack.listView-meta.xml b/nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsToBeSentToSlack.listView-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsToBeSentToSlack.listView-meta.xml rename to nebula-logger/plugins/slack/plugin/core/main/log-management/objects/Log__c/listViews/AllLogsToBeSentToSlack.listView-meta.xml diff --git a/nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin.cls b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin.cls similarity index 77% rename from nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin.cls rename to nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin.cls index 37c2228d6..5fc0dc945 100644 --- a/nebula-logger/plugins/Slack/plugin/slack/classes/SlackLoggerPlugin.cls +++ b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin.cls @@ -8,16 +8,14 @@ * @description Optional plugin that integrates with Slack to send alerts for important logs */ @SuppressWarnings('PMD.ExcessivePublicCount') -public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugin implements Queueable, Database.AllowsCallouts { +public without sharing class SlackLoggerPlugin implements LoggerPlugin.Triggerable, Queueable, Database.AllowsCallouts { @TestVisible - private static String endpoint = LoggerParameter.getString('SlackEndpoint', null); + private static final String ENDPOINT = LoggerParameter.getString('SlackEndpoint', null); @TestVisible - private static LoggingLevel notificationLoggingLevel = Logger.getLoggingLevel(LoggerParameter.getString('SlackNotificationLoggingLevel', null)); + private static final LoggingLevel NOTIFICATION_LOGGING_LEVEL = Logger.getLoggingLevel(LoggerParameter.getString('SlackNotificationLoggingLevel', null)); private List logs; - // Constructors - /** * @description Default constructor */ @@ -30,27 +28,20 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi this.logs = unsentLogs; } - // Instance methods - /** - * @description Handles the queuable execute logic. - * @param triggerOperationType Defines the trigger operation type - * @param triggerNew contains a list of the new trigger context / records. - * @param triggerNewMap contains a map of ids to new trigger records. - * @param triggerOld old contains a list of the old trigger context / records. - * @param triggerOldMap contains a map of ids to old trigger records. + * @description Handles the integration with Slack. This method is automatically called by Nebula Logger's plugin framework. + * @param configuration The instance of `LoggerPlugin__mdt` configured for this specific plugin + * @param input The instance of `LoggerTriggerableContext`, provided by the logging system */ @SuppressWarnings('PMD.ExcessiveParameterList') - public override void execute( - TriggerOperation triggerOperationType, - List triggerNew, - Map triggerNewMap, - List triggerOld, - Map triggerOldMap - ) { - this.logs = (List) triggerNew; - - switch on triggerOperationType { + public void execute(LoggerPlugin__mdt configuration, LoggerTriggerableContext input) { + if (input.sobjectType != Schema.Log__c.SObjectType || String.isBlank(ENDPOINT) == true) { + return; + } + + this.logs = (List) input.triggerNew; + + switch on input.triggerOperationType { when BEFORE_INSERT, BEFORE_UPDATE { this.flagLogsForSlackNotification(); } @@ -66,7 +57,7 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi */ @SuppressWarnings('PMD.AvoidDebugStatements') public void execute(System.QueueableContext queueableContext) { - // SInce this runs in an async context, requery the logs just in case any field values have changed + // Since this runs in an async context, requery the logs just in case any field values have changed this.requeryLogs(); if (this.logs.isEmpty() == true) { @@ -90,15 +81,20 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi // 'Short' is a reserved word in Apex, but used in Slack's API, so the conversion happens in JSON String notificationJson = JSON.serialize(notification).replace('"isShort"', '"short"'); request.setBody(notificationJson); + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.finest('Sending log entries to Slack endpoint').setHttpRequestDetails(request); + } HttpResponse response = new Http().send(request); - System.debug(notificationLoggingLevel, 'response.getStatusCode()==' + response.getStatusCode()); - System.debug(notificationLoggingLevel, 'response.getStatus()==' + response.getStatus()); + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.finest('Sent log entries to Slack endpoint').setHttpResponseDetails(response); + } log.SlackNotificationDate__c = System.now(); sentLogs.add(log); } } + Logger.saveLog(); update sentLogs; // If any logs couldn't be sent due to governor limits, start a new instance of the job @@ -108,12 +104,12 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi } private void flagLogsForSlackNotification() { - if (notificationLoggingLevel == null) { + if (NOTIFICATION_LOGGING_LEVEL == null) { return; } for (Log__c log : this.logs) { - if (log.MaxLogEntryLoggingLevelOrdinal__c >= notificationLoggingLevel.ordinal()) { + if (log.MaxLogEntryLoggingLevelOrdinal__c >= NOTIFICATION_LOGGING_LEVEL.ordinal()) { log.SendSlackNotification__c = true; } } @@ -161,33 +157,35 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi TotalWARNLogEntries__c, TransactionId__c, ( - SELECT Id, LoggingLevel__c, Message__c + SELECT Id, LoggingLevel__c, Message__c, ExceptionStackTrace__c, StackTrace__c FROM LogEntries__r - WHERE LoggingLevelOrdinal__c >= :notificationLoggingLevel.ordinal() + WHERE LoggingLevelOrdinal__c >= :NOTIFICATION_LOGGING_LEVEL.ordinal() ORDER BY Timestamp__c DESC LIMIT 1 ) FROM Log__c WHERE Id IN :this.logs - AND MaxLogEntryLoggingLevelOrdinal__c >= :notificationLoggingLevel.ordinal() + AND MaxLogEntryLoggingLevelOrdinal__c >= :NOTIFICATION_LOGGING_LEVEL.ordinal() AND SendSlackNotification__c = TRUE AND SlackNotificationDate__c = NULL ]; } - @SuppressWarnings('PMD.AvoidDebugStatements') private HttpRequest createSlackHttpRequest() { - System.debug(notificationLoggingLevel, 'endpoint==' + endpoint); - HttpRequest request = new HttpRequest(); - request.setEndpoint(endpoint); + request.setEndpoint(ENDPOINT); request.setMethod('POST'); request.setHeader('Content-Type', 'application/json'); + if (LoggerParameter.ENABLE_SYSTEM_MESSAGES == true) { + Logger.finest('Created Slack HTTP Request').setHttpRequestDetails(request); + } + return request; } + @SuppressWarnings('PMD.NcssMethodCount') private LogDto convertLog(Log__c log) { LogEntry__c lastLogEntry = log.LogEntries__r.get(0); String messageText = 'Last Log Entry Message' + '\n`' + lastLogEntry.LoggingLevel__c + ': ' + lastLogEntry.Message__c + '`'; @@ -201,6 +199,20 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi notification.title = log.Name; notification.title_link = Url.getSalesforceBaseUrl().toExternalForm() + '/' + log.Id; + if (String.isNotBlank(lastLogEntry.ExceptionStackTrace__c) == true) { + FieldDto exceptionStackTraceField = new FieldDto(); + exceptionStackTraceField.isShort = false; + exceptionStackTraceField.title = Schema.LogEntry__c.ExceptionStackTrace__c.getDescribe().getLabel(); + exceptionStackTraceField.value = '`' + lastLogEntry.ExceptionStackTrace__c + '`'; + notification.fields.add(exceptionStackTraceField); + } + + FieldDto stackTraceField = new FieldDto(); + stackTraceField.isShort = false; + stackTraceField.title = Schema.LogEntry__c.StackTrace__c.getDescribe().getLabel(); + stackTraceField.value = '`' + lastLogEntry.StackTrace__c + '`'; + notification.fields.add(stackTraceField); + // TODO: switch to dynamically creating Slack DTO fields based on a new `Log__c` field set parameter FieldDto startTimeField = new FieldDto(); startTimeField.isShort = true; @@ -226,6 +238,8 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi totalWARNEntriesField.value = '`' + String.valueOf(log.TotalWARNLogEntries__c) + '`'; notification.fields.add(totalWARNEntriesField); + // TODO Add Scenario__c field + String logOwnerType = log.OwnerId.getSObjectType().getDescribe().getName(); FieldDto logOwnerNameField = new FieldDto(); logOwnerNameField.isShort = true; @@ -263,20 +277,7 @@ public without sharing class SlackLoggerPlugin extends LoggerSObjectHandlerPlugi orgApiVersion.value = '`' + log.ApiVersion__c + '`'; notification.fields.add(orgApiVersion); - List topicNames = new List(); - // TODO - dynamically get either LogEntryTag__c or TopicAssignment (based on handler parameter) - for (TopicAssignment topicAssignment : log.TopicAssignments) { - topicNames.add(topicAssignment.Topic.Name); - } - topicNames.sort(); - - if (topicNames.isEmpty() == false) { - FieldDto topicsField = new FieldDto(); - topicsField.isShort = false; - topicsField.title = 'Topics'; - topicsField.value = '`' + String.join(topicNames, '`, `') + '`'; - notification.fields.add(topicsField); - } + // TODO Possible include to LogEntryTag__c (tag names) return notification; } diff --git a/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin.cls-meta.xml b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin.cls-meta.xml new file mode 100644 index 000000000..871a8cfea --- /dev/null +++ b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin.cls-meta.xml @@ -0,0 +1,5 @@ + + + 53.0 + Active + diff --git a/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls new file mode 100644 index 000000000..555548494 --- /dev/null +++ b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls @@ -0,0 +1,229 @@ +//------------------------------------------------------------------------------------------------// +// This file is part of the Nebula Logger project, released under the MIT License. // +// See LICENSE file or go to https://github.com/jongpie/NebulaLogger for full license details. // +//------------------------------------------------------------------------------------------------// + +@SuppressWarnings('PMD.ApexDoc, PMD.MethodNamingConventions') +@IsTest(IsParallel=true) +private class SlackLoggerPlugin_Tests { + @IsTest + static void it_should_not_push_log_when_logging_level_is_not_met() { + verifyLogEntryCountEquals(0); + Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), SendSlackNotification__c = false, TransactionId__c = '1234'); + insert log; + LoggingLevel logEntryLoggingLevel = LoggingLevel.WARN; + LogEntry__c logEntry = new LogEntry__c( + ExceptionStackTrace__c = 'Some exception stack trace', + Log__c = log.Id, + LoggingLevel__c = logEntryLoggingLevel.name(), + LoggingLevelOrdinal__c = logEntryLoggingLevel.ordinal(), + StackTrace__c = 'A stack trace', + Timestamp__c = System.now() + ); + insert logEntry; + verifyLogEntryCountEquals(1); + List logs = queryLogs(logEntryLoggingLevel); + System.assertEquals(1, logs.size(), 'Logs size did not match expected value of 1.'); + log = logs.get(0); + System.assertEquals(1, log.LogEntries__r.size(), 'Log entries size was not equal to 1.'); + System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to true.'); + System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); + Test.startTest(); + LoggerMockDataCreator.MockHttpCallout calloutMock = LoggerMockDataCreator.createHttpCallout().setStatusCode(200); + Test.setMock(HttpCalloutMock.class, calloutMock); + // Load the mock configurations - the plugin framework won't load actual CMDT records during tests + LoggingLevel slackLoggingLevel = LoggingLevel.ERROR; + System.assert(logEntryLoggingLevel.ordinal() < slackLoggingLevel.ordinal(), 'Slack logging level ordinal was incorrect.'); + mockConfigurations(slackLoggingLevel); + System.assert(logEntryLoggingLevel.ordinal() < SlackLoggerPlugin.NOTIFICATION_LOGGING_LEVEL.ordinal(), 'Slack logging level ordinal was incorrect.'); + LoggerSObjectHandler.shouldExecute(true); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + + // Update the records to trigger the handler framework, which will then run the Slack plugin + update log; + + // Verify that the internal queueable job has been enqueued + System.assertEquals(0, Limits.getAsyncCalls(), 'The queueable job should not have been enqueued'); + // Stop the test so the internal queueable job runs + Test.stopTest(); + log = queryLogs(logEntryLoggingLevel).get(0); + System.assertEquals(1, log.LogEntries__r.size(), 'Log entries size was not equal to 1.'); + System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification incorrectly set to true.'); + System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); + } + + @IsTest + static void it_should_push_log_when_logging_level_is_met_for_error() { + verifyLogEntryCountEquals(0); + LoggerSObjectHandler.shouldExecute(false); + Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), SendSlackNotification__c = false, TransactionId__c = '1234'); + insert log; + LoggingLevel logEntryLoggingLevel = LoggingLevel.ERROR; + LogEntry__c logEntry = new LogEntry__c( + ExceptionStackTrace__c = 'Some exception stack trace', + Log__c = log.Id, + LoggingLevel__c = logEntryLoggingLevel.name(), + LoggingLevelOrdinal__c = logEntryLoggingLevel.ordinal(), + StackTrace__c = 'A stack trace', + Timestamp__c = System.now() + ); + insert logEntry; + verifyLogEntryCountEquals(1); + List logs = queryLogs(logEntryLoggingLevel); + System.assertEquals(1, logs.size(), 'Logs size did not match expected value of 1.'); + log = logs.get(0); + System.assertEquals(1, log.LogEntries__r.size(), 'Log entries did not match the expected count of 1.'); + System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to true.'); + System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); + Test.startTest(); + LoggerMockDataCreator.MockHttpCallout calloutMock = LoggerMockDataCreator.createHttpCallout().setStatusCode(200); + Test.setMock(HttpCalloutMock.class, calloutMock); + // Load the mock configurations - the plugin framework won't load actual CMDT records during tests + mockConfigurations(logEntryLoggingLevel); + System.assert( + logEntryLoggingLevel.ordinal() >= SlackLoggerPlugin.NOTIFICATION_LOGGING_LEVEL.ordinal(), + 'The notification logging level ordinal was incorrect.' + ); + LoggerSObjectHandler.shouldExecute(true); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + + // Update the records to trigger the handler framework, which will then run the Slack plugin + update log; + + // Verify that the internal queueable job has been enqueued + System.assertEquals(1, Limits.getAsyncCalls(), 'The queueable job should have been enqueued'); + // Stop the test so the internal queueable job runs + Test.stopTest(); + log = [SELECT Id, MaxLogEntryLoggingLevelOrdinal__c, SendSlackNotification__c, SlackNotificationDate__c FROM Log__c]; + System.assertEquals(true, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to false.'); + System.assertNotEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was null.'); + System.assertEquals(System.today(), log.SlackNotificationDate__c.date(), 'SlackNotificationDate was not set to TODAY.'); + } + + @IsTest + static void it_should_push_log_when_logging_level_is_met_for_warn() { + verifyLogEntryCountEquals(0); + LoggerSObjectHandler.shouldExecute(false); + Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), SendSlackNotification__c = false, TransactionId__c = '1234'); + insert log; + LoggingLevel logEntryLoggingLevel = LoggingLevel.WARN; + LogEntry__c logEntry = new LogEntry__c( + ExceptionStackTrace__c = 'Some exception stack trace', + Log__c = log.Id, + LoggingLevel__c = logEntryLoggingLevel.name(), + LoggingLevelOrdinal__c = logEntryLoggingLevel.ordinal(), + StackTrace__c = 'A stack trace', + Timestamp__c = System.now() + ); + insert logEntry; + verifyLogEntryCountEquals(1); + List logs = queryLogs(logEntryLoggingLevel); + System.assertEquals(1, logs.size(), 'Logs size did not match expected value of 1.'); + log = logs.get(0); + System.assertEquals(1, log.LogEntries__r.size(), 'Log entries did not match the expected count of 1.'); + System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to true.'); + System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); + Test.startTest(); + LoggerMockDataCreator.MockHttpCallout calloutMock = LoggerMockDataCreator.createHttpCallout().setStatusCode(200); + Test.setMock(HttpCalloutMock.class, calloutMock); + // Load the mock configurations - the plugin framework won't load actual CMDT records during tests + mockConfigurations(logEntryLoggingLevel); + System.assert( + logEntryLoggingLevel.ordinal() >= SlackLoggerPlugin.NOTIFICATION_LOGGING_LEVEL.ordinal(), + 'The notification logging level ordinal was incorrect.' + ); + LoggerSObjectHandler.shouldExecute(true); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + + // Update the records to trigger the handler framework, which will then run the Slack plugin + update log; + + // Verify that the internal queueable job has been enqueued + System.assertEquals(1, Limits.getAsyncCalls(), 'The queueable job should have been enqueued'); + // Stop the test so the internal queueable job runs + Test.stopTest(); + log = [SELECT Id, MaxLogEntryLoggingLevelOrdinal__c, SendSlackNotification__c, SlackNotificationDate__c FROM Log__c]; + System.assertEquals(true, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to false.'); + System.assertNotEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was null.'); + System.assertEquals(System.today(), log.SlackNotificationDate__c.date(), 'SlackNotificationDate was not set to TODAY.'); + } + + @IsTest + static void it_should_push_log_when_logging_level_is_met_for_info() { + verifyLogEntryCountEquals(0); + LoggerSObjectHandler.shouldExecute(false); + Log__c log = new Log__c(LoggedBy__c = UserInfo.getUserId(), SendSlackNotification__c = false, TransactionId__c = '1234'); + insert log; + LoggingLevel logEntryLoggingLevel = LoggingLevel.INFO; + LogEntry__c logEntry = new LogEntry__c( + ExceptionStackTrace__c = 'Some exception stack trace', + Log__c = log.Id, + LoggingLevel__c = logEntryLoggingLevel.name(), + LoggingLevelOrdinal__c = logEntryLoggingLevel.ordinal(), + StackTrace__c = 'A stack trace', + Timestamp__c = System.now() + ); + insert logEntry; + verifyLogEntryCountEquals(1); + List logs = queryLogs(logEntryLoggingLevel); + System.assertEquals(1, logs.size(), 'Logs size did not match expected value of 1.'); + log = logs.get(0); + System.assertEquals(1, log.LogEntries__r.size(), 'Log entries did not match the expected count of 1.'); + System.assertEquals(false, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to true.'); + System.assertEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was not null.'); + Test.startTest(); + LoggerMockDataCreator.MockHttpCallout calloutMock = LoggerMockDataCreator.createHttpCallout().setStatusCode(200); + Test.setMock(HttpCalloutMock.class, calloutMock); + // Load the mock configurations - the plugin framework won't load actual CMDT records during tests + mockConfigurations(logEntryLoggingLevel); + System.assert( + logEntryLoggingLevel.ordinal() >= SlackLoggerPlugin.NOTIFICATION_LOGGING_LEVEL.ordinal(), + 'The notification logging level ordinal was incorrect.' + ); + LoggerSObjectHandler.shouldExecute(true); + LoggerTestConfigurator.setupMockSObjectHandlerConfigurations(); + + // Update the records to trigger the handler framework, which will then run the Slack plugin + update log; + + // Verify that the internal queueable job has been enqueued + System.assertEquals(1, Limits.getAsyncCalls(), 'The queueable job should have been enqueued'); + // Stop the test so the internal queueable job runs + Test.stopTest(); + log = [SELECT Id, MaxLogEntryLoggingLevelOrdinal__c, SendSlackNotification__c, SlackNotificationDate__c FROM Log__c]; + System.assertEquals(true, log.SendSlackNotification__c, 'SendSlackNotification was incorrectly set to false.'); + System.assertNotEquals(null, log.SlackNotificationDate__c, 'SlackNotificationDate was null.'); + System.assertEquals(System.today(), log.SlackNotificationDate__c.date(), 'SlackNotificationDate was not set to TODAY.'); + } + + static void mockConfigurations(LoggingLevel notificationLoggingLevel) { + LoggerTestConfigurator.setMock( + new LoggerPlugin__mdt(DeveloperName = 'SlackPlugin', IsEnabled__c = true, SObjectHandlerApexClass__c = SlackLoggerPlugin.class.getName()) + ); + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'SlackEndpoint', Value__c = 'https://fake.slack.com/')); + LoggerTestConfigurator.setMock(new LoggerParameter__mdt(DeveloperName = 'SlackNotificationLoggingLevel', Value__c = notificationLoggingLevel.name())); + } + + static void verifyLogEntryCountEquals(Integer expectedCount) { + Integer existingLogEntriesCount = [SELECT COUNT() FROM LogEntry__c]; + System.assertEquals(expectedCount, existingLogEntriesCount, 'Existing log entries did NOT match the expected count.'); + } + + static List queryLogs(LoggingLevel notificationLoggingLevel) { + return [ + SELECT + Id, + MaxLogEntryLoggingLevelOrdinal__c, + SendSlackNotification__c, + SlackNotificationDate__c, + ( + SELECT Id, LoggingLevel__c, Message__c, ExceptionStackTrace__c, StackTrace__c + FROM LogEntries__r + WHERE LoggingLevelOrdinal__c >= :notificationLoggingLevel.ordinal() + ORDER BY Timestamp__c DESC + LIMIT 1 + ) + FROM Log__c + ]; + } +} diff --git a/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls-meta.xml b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls-meta.xml new file mode 100644 index 000000000..871a8cfea --- /dev/null +++ b/nebula-logger/plugins/slack/plugin/slack/classes/SlackLoggerPlugin_Tests.cls-meta.xml @@ -0,0 +1,5 @@ + + + 53.0 + Active + diff --git a/nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerParameter.SlackEndpoint.md-meta.xml b/nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerParameter.SlackEndpoint.md-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerParameter.SlackEndpoint.md-meta.xml rename to nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerParameter.SlackEndpoint.md-meta.xml diff --git a/nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerParameter.SlackNotificationLoggingLevel.md-meta.xml b/nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerParameter.SlackNotificationLoggingLevel.md-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/slack/customMetadata/LoggerParameter.SlackNotificationLoggingLevel.md-meta.xml rename to nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerParameter.SlackNotificationLoggingLevel.md-meta.xml diff --git a/nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml b/nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml new file mode 100644 index 000000000..a3f0dadc5 --- /dev/null +++ b/nebula-logger/plugins/slack/plugin/slack/customMetadata/LoggerPlugin.Slack.md-meta.xml @@ -0,0 +1,51 @@ + + + + false + + BatchPurgerApexClass__c + + + + BatchPurgerExecutionOrder__c + + + + BatchPurgerFlowName__c + + + + Description__c + Adds a Slack integration for Nebula Logger. + +Any logs with MaxLogEntryLoggingLevelOrdinal__c >= the parameter 'SlackLoggingLevelThreshold' will send a notification to Slack + + + IsEnabled__c + true + + + Link__c + https://github.com/jongpie/NebulaLogger/tree/main/nebula-logger/plugins/Slack + + + SObjectHandlerApexClass__c + SlackLoggerPlugin + + + SObjectHandlerExecutionOrder__c + + + + SObjectHandlerFlowName__c + + + + VersionNumber__c + v0.9.0 + + diff --git a/nebula-logger/plugins/Slack/plugin/slack/permissionsets/LoggerSlackPluginAdmin.permissionset-meta.xml b/nebula-logger/plugins/slack/plugin/slack/permissionsets/LoggerSlackPluginAdmin.permissionset-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/slack/permissionsets/LoggerSlackPluginAdmin.permissionset-meta.xml rename to nebula-logger/plugins/slack/plugin/slack/permissionsets/LoggerSlackPluginAdmin.permissionset-meta.xml diff --git a/nebula-logger/plugins/Slack/plugin/slack/remoteSiteSettings/Slack.remoteSite-meta.xml b/nebula-logger/plugins/slack/plugin/slack/remoteSiteSettings/Slack.remoteSite-meta.xml similarity index 100% rename from nebula-logger/plugins/Slack/plugin/slack/remoteSiteSettings/Slack.remoteSite-meta.xml rename to nebula-logger/plugins/slack/plugin/slack/remoteSiteSettings/Slack.remoteSite-meta.xml diff --git a/nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls b/nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls new file mode 100644 index 000000000..0bab3522b --- /dev/null +++ b/nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls @@ -0,0 +1,22 @@ +@SuppressWarnings('PMD.ApexDoc') +public with sharing class ExampleBigObjectDataGenerator implements System.Queueable { + private Integer targetRecordCountToCreate; + + public ExampleBigObjectDataGenerator(Integer targetRecordCountToCreate) { + this.targetRecordCountToCreate = targetRecordCountToCreate; + } + + public void execute(System.QueueableContext queueableContext) { + // Integer target = targetRecordCountToCreate <= 200 ? 200 : targetRecordCountToCreate; + for (Integer i = 0; i < 150; i++) { + Logger.info('hello, testing some bulk logging into big object via queueable job, record index is: ' + i); + this.targetRecordCountToCreate--; + } + Logger.getUserSettings().DefaultSaveMethod__c = 'BIG_OBJECT_IMMEDIATE'; + Logger.saveLog(); + + if (this.targetRecordCountToCreate > 0) { + System.enqueueJob(new ExampleBigObjectDataGenerator(this.targetRecordCountToCreate)); + } + } +} diff --git a/nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls-meta.xml b/nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls-meta.xml new file mode 100644 index 000000000..891916bb0 --- /dev/null +++ b/nebula-logger/recipes/classes/ExampleBigObjectDataGenerator.cls-meta.xml @@ -0,0 +1,5 @@ + + + 54.0 + Active + diff --git a/nebula-logger/recipes/profiles/Admin.profile-meta.xml b/nebula-logger/recipes/profiles/Admin.profile-meta.xml index 697fc95eb..949410f39 100644 --- a/nebula-logger/recipes/profiles/Admin.profile-meta.xml +++ b/nebula-logger/recipes/profiles/Admin.profile-meta.xml @@ -2,22 +2,9 @@ LoggerConsole - true - true - - - LoggerRecipes false - true + false - - Account_Batch_Logger_Example - true - - - Account_Queueable_Logger_Example - true - ComponentLogger true @@ -26,10 +13,6 @@ ComponentLogger_Tests true - - ExampleClassWithLogging - true - ExampleInboundEmailHandler true @@ -83,17 +66,45 @@ true - LogEntryEventBuilder + LogBatchPurger_Tests_Integration true - LogEntryEventBuilder_Tests_Integration + LogEntryArchiveBuilder + true + + + LogEntryArchiveBuilder_Tests + true + + + LogEntryArchiveController + true + + + LogEntryArchiveController_Tests + true + + + LogEntryArchivePlugin + true + + + LogEntryArchivePlugin_Tests + true + + + LogEntryEventBuilder true LogEntryEventBuilder_Tests true + + LogEntryEventBuilder_Tests_Integration + true + LogEntryEventHandler true @@ -139,15 +150,15 @@ true - LogHandler_Tests_Flow + LogMassDeleteExtension true - LogMassDeleteExtension + LogMassDeleteExtension_Tests true - LogMassDeleteExtension_Tests + LogMassDeleteExtension_Tests_Integration true @@ -158,10 +169,26 @@ LogMessage_Tests true + + LogRetentionRulesPlugin + true + + + LogRetentionRulesPlugin_Tests + true + Logger true + + LoggerEmailSender + true + + + LoggerEmailSender_Tests + true + LoggerParameter true @@ -170,16 +197,40 @@ LoggerParameter_Tests true + + LoggerPlugin + true + + + LoggerPlugin_Tests + true + LoggerSObjectHandler true - LoggerSObjectHandlerPlugin + LoggerSObjectHandler_Tests + true + + + LoggerSObjectMetadata + true + + + LoggerSObjectMetadata_Tests true - LoggerSObjectHandlerPlugin_Tests + LoggerSettingsController + true + + + LoggerSettingsController_Tests + true + + + LoggerSettingsController_Tests_Security true @@ -191,7 +242,7 @@ true - LoggerTestUtils + LoggerMockDataCreator true @@ -229,717 +280,1222 @@ false false - LogEntryDataMaskRule__mdt.ApplyToMessage__c + LogEntryArchive__b.ApiVersion__c true false - LogEntryDataMaskRule__mdt.ApplyToRecordJson__c + LogEntryArchive__b.ComponentType__c true false - LogEntryDataMaskRule__mdt.IsEnabled__c + LogEntryArchive__b.DatabaseResultCollectionType__c true false - LogEntryEvent__e.ApiVersion__c + LogEntryArchive__b.DatabaseResultJson__c true false - LogEntryEvent__e.ComponentType__c + LogEntryArchive__b.DatabaseResultType__c true false - LogEntryEvent__e.DatabaseResultCollectionType__c + LogEntryArchive__b.EpochTimestamp__c true false - LogEntryEvent__e.DatabaseResultJson__c + LogEntryArchive__b.ExceptionMessage__c true false - LogEntryEvent__e.DatabaseResultType__c + LogEntryArchive__b.ExceptionStackTrace__c true false - LogEntryEvent__e.EpochTimestamp__c + LogEntryArchive__b.ExceptionType__c true false - LogEntryEvent__e.ExceptionMessage__c + LogEntryArchive__b.LimitsAggregateQueriesMax__c true false - LogEntryEvent__e.ExceptionStackTrace__c + LogEntryArchive__b.LimitsAggregateQueriesUsed__c true false - LogEntryEvent__e.ExceptionType__c + LogEntryArchive__b.LimitsAggregateQueryMax__c true false - LogEntryEvent__e.LimitsAggregateQueriesMax__c + LogEntryArchive__b.LimitsAsyncCallsMax__c true false - LogEntryEvent__e.LimitsAggregateQueriesUsed__c + LogEntryArchive__b.LimitsAsyncCallsUsed__c true false - LogEntryEvent__e.LimitsAggregateQueryMax__c + LogEntryArchive__b.LimitsCalloutsMax__c true false - LogEntryEvent__e.LimitsAsyncCallsMax__c + LogEntryArchive__b.LimitsCalloutsUsed__c true false - LogEntryEvent__e.LimitsAsyncCallsUsed__c + LogEntryArchive__b.LimitsCpuTimeMax__c true false - LogEntryEvent__e.LimitsCalloutsMax__c + LogEntryArchive__b.LimitsCpuTimeUsed__c true false - LogEntryEvent__e.LimitsCalloutsUsed__c + LogEntryArchive__b.LimitsDmlRowsMax__c true false - LogEntryEvent__e.LimitsCpuTimeMax__c + LogEntryArchive__b.LimitsDmlRowsUsed__c true false - LogEntryEvent__e.LimitsCpuTimeUsed__c + LogEntryArchive__b.LimitsDmlStatementsMax__c true false - LogEntryEvent__e.LimitsDmlRowsMax__c + LogEntryArchive__b.LimitsDmlStatementsUsed__c true false - LogEntryEvent__e.LimitsDmlRowsUsed__c + LogEntryArchive__b.LimitsEmailInvocationsMax__c true false - LogEntryEvent__e.LimitsDmlStatementsMax__c + LogEntryArchive__b.LimitsEmailInvocationsUsed__c true false - LogEntryEvent__e.LimitsDmlStatementsUsed__c + LogEntryArchive__b.LimitsFutureCallsMax__c true false - LogEntryEvent__e.LimitsEmailInvocationsMax__c + LogEntryArchive__b.LimitsFutureCallsUsed__c true false - LogEntryEvent__e.LimitsEmailInvocationsUsed__c + LogEntryArchive__b.LimitsHeapSizeMax__c true false - LogEntryEvent__e.LimitsFutureCallsMax__c + LogEntryArchive__b.LimitsHeapSizeUsed__c true false - LogEntryEvent__e.LimitsFutureCallsUsed__c + LogEntryArchive__b.LimitsMobilePushApexCallsMax__c true false - LogEntryEvent__e.LimitsHeapSizeMax__c + LogEntryArchive__b.LimitsMobilePushApexCallsUsed__c true false - LogEntryEvent__e.LimitsHeapSizeUsed__c + LogEntryArchive__b.LimitsPublishImmediateDmlStatementsMax__c true false - LogEntryEvent__e.LimitsMobilePushApexCallsMax__c + LogEntryArchive__b.LimitsPublishImmediateDmlStatementsUsed__c true false - LogEntryEvent__e.LimitsMobilePushApexCallsUsed__c + LogEntryArchive__b.LimitsQueueableJobsMax__c true false - LogEntryEvent__e.LimitsPublishImmediateDmlStatementsMax__c + LogEntryArchive__b.LimitsQueueableJobsUsed__c true false - LogEntryEvent__e.LimitsPublishImmediateDmlStatementsUsed__c + LogEntryArchive__b.LimitsSoqlQueriesMax__c true false - LogEntryEvent__e.LimitsQueueableJobsMax__c + LogEntryArchive__b.LimitsSoqlQueriesUsed__c true false - LogEntryEvent__e.LimitsQueueableJobsUsed__c + LogEntryArchive__b.LimitsSoqlQueryLocatorRowsMax__c true false - LogEntryEvent__e.LimitsSoqlQueriesMax__c + LogEntryArchive__b.LimitsSoqlQueryLocatorRowsUsed__c true false - LogEntryEvent__e.LimitsSoqlQueriesUsed__c + LogEntryArchive__b.LimitsSoqlQueryRowsMax__c true false - LogEntryEvent__e.LimitsSoqlQueryLocatorRowsMax__c + LogEntryArchive__b.LimitsSoqlQueryRowsUsed__c true false - LogEntryEvent__e.LimitsSoqlQueryLocatorRowsUsed__c + LogEntryArchive__b.LimitsSoslSearchesMax__c true false - LogEntryEvent__e.LimitsSoqlQueryRowsMax__c + LogEntryArchive__b.LimitsSoslSearchesUsed__c true false - LogEntryEvent__e.LimitsSoqlQueryRowsUsed__c + LogEntryArchive__b.Locale__c true false - LogEntryEvent__e.LimitsSoslSearchesMax__c + LogEntryArchive__b.LoggedById__c true false - LogEntryEvent__e.LimitsSoslSearchesUsed__c + LogEntryArchive__b.LoggedByUsername__c true false - LogEntryEvent__e.Locale__c + LogEntryArchive__b.LoggerVersionNumber__c true false - LogEntryEvent__e.LoggedById__c + LogEntryArchive__b.LoginApplication__c true false - LogEntryEvent__e.LoggedByUsername__c + LogEntryArchive__b.LoginBrowser__c true false - LogEntryEvent__e.LoggingLevelOrdinal__c + LogEntryArchive__b.LoginDomain__c true false - LogEntryEvent__e.LoggingLevel__c + LogEntryArchive__b.LoginHistoryId__c true false - LogEntryEvent__e.LoggerVersionNumber__c + LogEntryArchive__b.LoginPlatform__c true false - LogEntryEvent__e.LoginApplication__c + LogEntryArchive__b.LoginType__c true false - LogEntryEvent__e.LoginBrowser__c + LogEntryArchive__b.LogoutUrl__c true false - LogEntryEvent__e.LoginDomain__c + LogEntryArchive__b.Message__c true false - LogEntryEvent__e.LoginHistoryId__c + LogEntryArchive__b.NetworkId__c true false - LogEntryEvent__e.LoginPlatform__c + LogEntryArchive__b.NetworkLoginUrl__c true false - LogEntryEvent__e.LoginType__c + LogEntryArchive__b.NetworkLogoutUrl__c true false - LogEntryEvent__e.LogoutUrl__c + LogEntryArchive__b.NetworkName__c true false - LogEntryEvent__e.MessageMasked__c + LogEntryArchive__b.NetworkSelfRegistrationUrl__c true false - LogEntryEvent__e.MessageTruncated__c + LogEntryArchive__b.NetworkUrlPathPrefix__c true false - LogEntryEvent__e.Message__c + LogEntryArchive__b.OrganizationDomainUrl__c true false - LogEntryEvent__e.NetworkId__c + LogEntryArchive__b.OrganizationEnvironmentType__c true false - LogEntryEvent__e.NetworkLoginUrl__c + LogEntryArchive__b.OrganizationId__c true false - LogEntryEvent__e.NetworkLogoutUrl__c + LogEntryArchive__b.OrganizationInstanceName__c true false - LogEntryEvent__e.NetworkName__c + LogEntryArchive__b.OrganizationName__c true false - LogEntryEvent__e.NetworkSelfRegistrationUrl__c + LogEntryArchive__b.OrganizationNamespacePrefix__c true false - LogEntryEvent__e.NetworkUrlPathPrefix__c + LogEntryArchive__b.OrganizationType__c true false - LogEntryEvent__e.OrganizationDomainUrl__c + LogEntryArchive__b.OriginLocation__c true false - LogEntryEvent__e.OrganizationEnvironmentType__c + LogEntryArchive__b.OriginType__c true false - LogEntryEvent__e.OrganizationId__c + LogEntryArchive__b.ParentLogTransactionId__c true false - LogEntryEvent__e.OrganizationInstanceName__c + LogEntryArchive__b.ProfileId__c true false - LogEntryEvent__e.OrganizationName__c + LogEntryArchive__b.ProfileName__c true false - LogEntryEvent__e.OrganizationNamespacePrefix__c + LogEntryArchive__b.RecordCollectionType__c true false - LogEntryEvent__e.OrganizationType__c + LogEntryArchive__b.RecordId__c true false - LogEntryEvent__e.OriginLocation__c + LogEntryArchive__b.RecordJson__c true false - LogEntryEvent__e.OriginType__c + LogEntryArchive__b.RecordSObjectClassification__c true false - LogEntryEvent__e.ParentLogTransactionId__c + LogEntryArchive__b.RecordSObjectTypeNamespace__c true false - LogEntryEvent__e.ProfileId__c + LogEntryArchive__b.RecordSObjectType__c true false - LogEntryEvent__e.ProfileName__c + LogEntryArchive__b.Scenario__c true false - LogEntryEvent__e.RecordCollectionType__c + LogEntryArchive__b.SessionId__c true false - LogEntryEvent__e.RecordId__c + LogEntryArchive__b.SessionSecurityLevel__c true false - LogEntryEvent__e.RecordJsonMasked__c + LogEntryArchive__b.SessionType__c true false - LogEntryEvent__e.RecordJson__c + LogEntryArchive__b.SourceIp__c true false - LogEntryEvent__e.RecordSObjectClassification__c + LogEntryArchive__b.StackTrace__c true false - LogEntryEvent__e.RecordSObjectTypeNamespace__c + LogEntryArchive__b.SystemMode__c true false - LogEntryEvent__e.RecordSObjectType__c + LogEntryArchive__b.Tags__c true false - LogEntryEvent__e.SessionId__c + LogEntryArchive__b.ThemeDisplayed__c true false - LogEntryEvent__e.SessionSecurityLevel__c + LogEntryArchive__b.TimeZoneId__c true false - LogEntryEvent__e.SessionType__c + LogEntryArchive__b.TimeZoneName__c true false - LogEntryEvent__e.SourceIp__c + LogEntryArchive__b.TimestampString__c true false - LogEntryEvent__e.StackTrace__c + LogEntryArchive__b.TriggerOperationType__c true false - LogEntryEvent__e.SystemMode__c + LogEntryArchive__b.TriggerSObjectType__c true false - LogEntryEvent__e.Tags__c + LogEntryArchive__b.UserLicenseDefinitionKey__c true false - LogEntryEvent__e.ThemeDisplayed__c + LogEntryArchive__b.UserLicenseId__c true false - LogEntryEvent__e.TimeZoneId__c + LogEntryArchive__b.UserLicenseName__c true false - LogEntryEvent__e.TimeZoneName__c + LogEntryArchive__b.UserLoggingLevelOrdinal__c true false - LogEntryEvent__e.TimestampString__c + LogEntryArchive__b.UserLoggingLevel__c true false - LogEntryEvent__e.Topics__c + LogEntryArchive__b.UserRoleId__c true false - LogEntryEvent__e.TriggerIsExecuting__c + LogEntryArchive__b.UserRoleName__c true false - LogEntryEvent__e.TriggerOperationType__c + LogEntryArchive__b.UserType__c true false - LogEntryEvent__e.TriggerSObjectType__c + LogEntryDataMaskRule__mdt.ApplyToMessage__c true false - LogEntryEvent__e.UserLicenseDefinitionKey__c + LogEntryDataMaskRule__mdt.ApplyToRecordJson__c true false - LogEntryEvent__e.UserLicenseId__c + LogEntryDataMaskRule__mdt.IsEnabled__c true false - LogEntryEvent__e.UserLicenseName__c + LogEntryEvent__e.ApiVersion__c true false - LogEntryEvent__e.UserLoggingLevelOrdinal__c + LogEntryEvent__e.ComponentType__c true false - LogEntryEvent__e.UserLoggingLevel__c + LogEntryEvent__e.DatabaseResultCollectionType__c true false - LogEntryEvent__e.UserRoleId__c + LogEntryEvent__e.DatabaseResultJson__c true false - LogEntryEvent__e.UserRoleName__c + LogEntryEvent__e.DatabaseResultType__c true false - LogEntryEvent__e.UserType__c + LogEntryEvent__e.EpochTimestamp__c true false - LogEntryTagRule__mdt.ComparisonValue__c + LogEntryEvent__e.ExceptionMessage__c true false - LogEntryTagRule__mdt.IsEnabled__c + LogEntryEvent__e.ExceptionStackTrace__c true false - LogEntryTagRule__mdt.Tags__c + LogEntryEvent__e.ExceptionType__c true false - LogEntryTag__c.LogEntryOrigin__c + LogEntryEvent__e.LimitsAggregateQueriesMax__c true false - LogEntryTag__c.LogEntryTimestamp__c + LogEntryEvent__e.LimitsAggregateQueriesUsed__c true false - LogEntryTag__c.LogLink__c + LogEntryEvent__e.LimitsAggregateQueryMax__c true false - LogEntryTag__c.LoggedByUsernameLink__c + LogEntryEvent__e.LimitsAsyncCallsMax__c true false - LogEntryTag__c.UniqueId__c + LogEntryEvent__e.LimitsAsyncCallsUsed__c true false - LogEntry__c.ApexClassApiVersion__c + LogEntryEvent__e.LimitsCalloutsMax__c true false - LogEntry__c.ApexClassCreatedDate__c + LogEntryEvent__e.LimitsCalloutsUsed__c true false - LogEntry__c.ApexClassId__c + LogEntryEvent__e.LimitsCpuTimeMax__c true false - LogEntry__c.ApexClassLastModifiedDate__c + LogEntryEvent__e.LimitsCpuTimeUsed__c true false - LogEntry__c.ApexClassName__c + LogEntryEvent__e.LimitsDmlRowsMax__c true false - LogEntry__c.ApexInnerClassName__c + LogEntryEvent__e.LimitsDmlRowsUsed__c true false - LogEntry__c.ApexMethodName__c + LogEntryEvent__e.LimitsDmlStatementsMax__c true false - LogEntry__c.ComponentApiName__c + LogEntryEvent__e.LimitsDmlStatementsUsed__c true false - LogEntry__c.ComponentFunctionName__c + LogEntryEvent__e.LimitsEmailInvocationsMax__c true false - LogEntry__c.ComponentType__c + LogEntryEvent__e.LimitsEmailInvocationsUsed__c true false - LogEntry__c.DatabaseResultCollectionType__c + LogEntryEvent__e.LimitsFutureCallsMax__c true false - LogEntry__c.DatabaseResultJson__c + LogEntryEvent__e.LimitsFutureCallsUsed__c true false - LogEntry__c.DatabaseResultType__c + LogEntryEvent__e.LimitsHeapSizeMax__c true false - LogEntry__c.EpochTimestamp__c + LogEntryEvent__e.LimitsHeapSizeUsed__c true false - LogEntry__c.EventUuid__c + LogEntryEvent__e.LimitsMobilePushApexCallsMax__c true false - LogEntry__c.ExceptionMessage__c + LogEntryEvent__e.LimitsMobilePushApexCallsUsed__c true false - LogEntry__c.ExceptionStackTrace__c + LogEntryEvent__e.LimitsPublishImmediateDmlStatementsMax__c true false - LogEntry__c.ExceptionType__c + LogEntryEvent__e.LimitsPublishImmediateDmlStatementsUsed__c true false - LogEntry__c.FlowActiveVersionId__c + LogEntryEvent__e.LimitsQueueableJobsMax__c true false - LogEntry__c.FlowDescription__c + LogEntryEvent__e.LimitsQueueableJobsUsed__c true false - LogEntry__c.FlowDurableId__c + LogEntryEvent__e.LimitsSoqlQueriesMax__c true false - LogEntry__c.FlowLabel__c + LogEntryEvent__e.LimitsSoqlQueriesUsed__c true false - LogEntry__c.FlowLastModifiedByName__c + LogEntryEvent__e.LimitsSoqlQueryLocatorRowsMax__c true false - LogEntry__c.FlowLastModifiedDate__c + LogEntryEvent__e.LimitsSoqlQueryLocatorRowsUsed__c true false - LogEntry__c.FlowProcessType__c + LogEntryEvent__e.LimitsSoqlQueryRowsMax__c true false - LogEntry__c.FlowTriggerType__c + LogEntryEvent__e.LimitsSoqlQueryRowsUsed__c + true + + + false + LogEntryEvent__e.LimitsSoslSearchesMax__c + true + + + false + LogEntryEvent__e.LimitsSoslSearchesUsed__c + true + + + false + LogEntryEvent__e.Locale__c + true + + + false + LogEntryEvent__e.LoggedById__c + true + + + false + LogEntryEvent__e.LoggedByUsername__c + true + + + false + LogEntryEvent__e.LoggerVersionNumber__c + true + + + false + LogEntryEvent__e.LoggingLevelOrdinal__c + true + + + false + LogEntryEvent__e.LoggingLevel__c + true + + + false + LogEntryEvent__e.LoginApplication__c + true + + + false + LogEntryEvent__e.LoginBrowser__c + true + + + false + LogEntryEvent__e.LoginDomain__c + true + + + false + LogEntryEvent__e.LoginHistoryId__c + true + + + false + LogEntryEvent__e.LoginPlatform__c + true + + + false + LogEntryEvent__e.LoginType__c + true + + + false + LogEntryEvent__e.LogoutUrl__c + true + + + false + LogEntryEvent__e.MessageMasked__c + true + + + false + LogEntryEvent__e.MessageTruncated__c + true + + + false + LogEntryEvent__e.Message__c + true + + + false + LogEntryEvent__e.NetworkId__c + true + + + false + LogEntryEvent__e.NetworkLoginUrl__c + true + + + false + LogEntryEvent__e.NetworkLogoutUrl__c + true + + + false + LogEntryEvent__e.NetworkName__c + true + + + false + LogEntryEvent__e.NetworkSelfRegistrationUrl__c + true + + + false + LogEntryEvent__e.NetworkUrlPathPrefix__c + true + + + false + LogEntryEvent__e.OrganizationDomainUrl__c + true + + + false + LogEntryEvent__e.OrganizationEnvironmentType__c + true + + + false + LogEntryEvent__e.OrganizationId__c + true + + + false + LogEntryEvent__e.OrganizationInstanceName__c + true + + + false + LogEntryEvent__e.OrganizationName__c + true + + + false + LogEntryEvent__e.OrganizationNamespacePrefix__c + true + + + false + LogEntryEvent__e.OrganizationType__c + true + + + false + LogEntryEvent__e.OriginLocation__c + true + + + false + LogEntryEvent__e.OriginType__c + true + + + false + LogEntryEvent__e.ParentLogTransactionId__c + true + + + false + LogEntryEvent__e.ProfileId__c + true + + + false + LogEntryEvent__e.ProfileName__c + true + + + false + LogEntryEvent__e.RecordCollectionType__c + true + + + false + LogEntryEvent__e.RecordId__c + true + + + false + LogEntryEvent__e.RecordJsonMasked__c + true + + + false + LogEntryEvent__e.RecordJson__c + true + + + false + LogEntryEvent__e.RecordSObjectClassification__c + true + + + false + LogEntryEvent__e.RecordSObjectTypeNamespace__c + true + + + false + LogEntryEvent__e.RecordSObjectType__c + true + + + false + LogEntryEvent__e.Scenario__c + true + + + false + LogEntryEvent__e.SessionId__c + true + + + false + LogEntryEvent__e.SessionSecurityLevel__c + true + + + false + LogEntryEvent__e.SessionType__c + true + + + false + LogEntryEvent__e.SourceIp__c + true + + + false + LogEntryEvent__e.StackTrace__c + true + + + false + LogEntryEvent__e.SystemMode__c + true + + + false + LogEntryEvent__e.Tags__c + true + + + false + LogEntryEvent__e.ThemeDisplayed__c + true + + + false + LogEntryEvent__e.TimeZoneId__c + true + + + false + LogEntryEvent__e.TimeZoneName__c + true + + + false + LogEntryEvent__e.TimestampString__c + true + + + false + LogEntryEvent__e.Topics__c + true + + + false + LogEntryEvent__e.TriggerIsExecuting__c + true + + + false + LogEntryEvent__e.TriggerOperationType__c + true + + + false + LogEntryEvent__e.TriggerSObjectType__c + true + + + false + LogEntryEvent__e.UserLicenseDefinitionKey__c + true + + + false + LogEntryEvent__e.UserLicenseId__c + true + + + false + LogEntryEvent__e.UserLicenseName__c + true + + + false + LogEntryEvent__e.UserLoggingLevelOrdinal__c + true + + + false + LogEntryEvent__e.UserLoggingLevel__c + true + + + false + LogEntryEvent__e.UserRoleId__c + true + + + false + LogEntryEvent__e.UserRoleName__c + true + + + false + LogEntryEvent__e.UserType__c + true + + + false + LogEntryTagRule__mdt.ComparisonValue__c + true + + + false + LogEntryTagRule__mdt.IsEnabled__c + true + + + false + LogEntryTagRule__mdt.Tags__c + true + + + false + LogEntryTag__c.LogEntryOrigin__c + true + + + false + LogEntryTag__c.LogEntryTimestamp__c + true + + + false + LogEntryTag__c.LogLink__c + true + + + false + LogEntryTag__c.LoggedByUsernameLink__c + true + + + false + LogEntryTag__c.UniqueId__c + true + + + false + LogEntry__c.ApexClassApiVersion__c + true + + + false + LogEntry__c.ApexClassCreatedDate__c + true + + + false + LogEntry__c.ApexClassId__c + true + + + false + LogEntry__c.ApexClassLastModifiedDate__c + true + + + false + LogEntry__c.ApexClassName__c + true + + + false + LogEntry__c.ApexInnerClassName__c + true + + + false + LogEntry__c.ApexMethodName__c + true + + + false + LogEntry__c.ComponentApiName__c + true + + + false + LogEntry__c.ComponentFunctionName__c + true + + + false + LogEntry__c.ComponentType__c + true + + + false + LogEntry__c.DatabaseResultCollectionType__c + true + + + false + LogEntry__c.DatabaseResultJson__c + true + + + false + LogEntry__c.DatabaseResultType__c + true + + + false + LogEntry__c.EpochTimestamp__c + true + + + false + LogEntry__c.EventUuid__c + true + + + false + LogEntry__c.ExceptionMessage__c + true + + + false + LogEntry__c.ExceptionStackTrace__c + true + + + false + LogEntry__c.ExceptionType__c + true + + + false + LogEntry__c.FlowActiveVersionId__c + true + + + false + LogEntry__c.FlowDescription__c + true + + + false + LogEntry__c.FlowDurableId__c + true + + + false + LogEntry__c.FlowLabel__c + true + + + false + LogEntry__c.FlowLastModifiedByName__c + true + + + false + LogEntry__c.FlowLastModifiedDate__c + true + + + false + LogEntry__c.FlowProcessType__c + true + + + false + LogEntry__c.FlowTriggerType__c true @@ -1274,97 +1830,137 @@ false - LogEntry__c.OriginType__c + LogEntry__c.OriginType__c + true + + + false + LogEntry__c.Origin__c + true + + + false + LogEntry__c.RecordCollectionType__c + true + + + false + LogEntry__c.RecordDetailedLink__c + true + + + false + LogEntry__c.RecordId__c + true + + + false + LogEntry__c.RecordJsonMasked__c + true + + + false + LogEntry__c.RecordJson__c + true + + + false + LogEntry__c.RecordLink__c + true + + + false + LogEntry__c.RecordName__c true false - LogEntry__c.Origin__c + LogEntry__c.RecordSObjectClassification__c true false - LogEntry__c.RecordCollectionType__c + LogEntry__c.RecordSObjectTypeNamespace__c true false - LogEntry__c.RecordDetailedLink__c + LogEntry__c.RecordSObjectType__c true false - LogEntry__c.RecordId__c + LogEntry__c.StackTrace__c true false - LogEntry__c.RecordJsonMasked__c + LogEntry__c.Timestamp__c true false - LogEntry__c.RecordJson__c + LogEntry__c.TransactionEntryNumber__c true false - LogEntry__c.RecordLink__c + LogEntry__c.TriggerIsExecuting__c true false - LogEntry__c.RecordName__c + LogEntry__c.TriggerOperationType__c true false - LogEntry__c.RecordSObjectClassification__c + LogEntry__c.TriggerSObjectType__c true false - LogEntry__c.RecordSObjectTypeNamespace__c + LogEntry__c.Trigger__c true false - LogEntry__c.RecordSObjectType__c + LogRetentionRuleCondition__mdt.SortOrder__c true false - LogEntry__c.StackTrace__c + LogRetentionRuleCondition__mdt.Value__c true false - LogEntry__c.Timestamp__c + LogRetentionRule__mdt.CustomConditionLogic__c true false - LogEntry__c.TransactionEntryNumber__c + LogRetentionRule__mdt.ExecutionOrder__c true false - LogEntry__c.TriggerIsExecuting__c + LogRetentionRule__mdt.IsEnabled__c true false - LogEntry__c.TriggerOperationType__c + LogRetentionRule__mdt.NumberOfDaysToRetainLogs__c true false - LogEntry__c.TriggerSObjectType__c + LogScenarioRule__mdt.IsEnabled__c true false - LogEntry__c.Trigger__c + LogScenarioRule__mdt.NumberOfDaysToRetainLogs__c true @@ -1442,6 +2038,11 @@ Log__c.LogEntriesSummary__c true + + false + Log__c.LogPurgeAction__c + true + false Log__c.LogRetentionDate__c @@ -1784,12 +2385,22 @@ false - LoggerPlugin__mdt.Description__c + LoggerPlugin__mdt.BatchPurgerApexClass__c + true + + + false + LoggerPlugin__mdt.BatchPurgerExecutionOrder__c + true + + + false + LoggerPlugin__mdt.BatchPurgerFlowName__c true false - LoggerPlugin__mdt.ExecutionOrder__c + LoggerPlugin__mdt.Description__c true @@ -1797,6 +2408,31 @@ LoggerPlugin__mdt.IsEnabled__c true + + false + LoggerPlugin__mdt.Link__c + true + + + false + LoggerPlugin__mdt.SObjectHandlerApexClass__c + true + + + false + LoggerPlugin__mdt.SObjectHandlerExecutionOrder__c + true + + + false + LoggerPlugin__mdt.SObjectHandlerFlowName__c + true + + + false + LoggerPlugin__mdt.VersionNumber__c + true + false LoggerTag__c.TotalLogEntries__c @@ -1819,6 +2455,15 @@ LogEntry__c-Log Entry Layout + + LogRetentionRuleCondition__mdt-Log Retention Rule Condition Layout + + + LogRetentionRule__mdt-Log Retention Rule Layout + + + LogScenarioRule__mdt-Log Scenario Rule Layout + LogStatus__mdt-Log Status Layout @@ -1834,6 +2479,15 @@ LoggerTag__c-Logger Tag Layout + + true + true + true + true + true + LogEntryArchive__b + true + true true @@ -1883,32 +2537,729 @@ LogMassDelete true + + LogEntryArchives + Hidden + LogEntryEventStream - DefaultOn + Hidden LogEntryTag__c - DefaultOn + Hidden LogEntry__c - DefaultOn + Hidden Log__c - DefaultOn + Hidden - LoggerAuraDemo - DefaultOn + LoggerSettings + Hidden LoggerTag__c - DefaultOn + Hidden - - Logger_lwc_demo - DefaultOn - - + Salesforce + + true + ActivateContract + + + true + ActivateOrder + + + true + ActivitiesAccess + + + true + AddDirectMessageMembers + + + true + AllowUniversalSearch + + + true + AllowViewKnowledge + + + true + ApexRestServices + + + true + ApiEnabled + + + true + ApproveContract + + + true + AssignPermissionSets + + + true + AssignTopics + + + true + AuthorApex + + + true + BulkMacrosAllowed + + + true + CanInsertFeedSystemFields + + + true + CanUseNewDashboardBuilder + + + true + CanVerifyComment + + + true + ChangeDashboardColors + + + true + ChatterEditOwnPost + + + true + ChatterEditOwnRecordPost + + + true + ChatterFileLink + + + true + ChatterInternalUser + + + true + ChatterInviteExternalUsers + + + true + ChatterOwnGroups + + + true + ConnectOrgToEnvironmentHub + + + true + ConsentApiUpdate + + + true + ContentAdministrator + + + true + ContentWorkspaces + + + true + ConvertLeads + + + true + CreateCustomizeDashboards + + + true + CreateCustomizeFilters + + + true + CreateCustomizeReports + + + true + CreateDashboardFolders + + + true + CreateLtngTempFolder + + + true + CreateReportFolders + + + true + CreateTopics + + + true + CreateWorkspaces + + + true + CustomizeApplication + + + true + DataExport + + + true + DelegatedTwoFactor + + + true + DeleteActivatedContract + + + true + DeleteTopics + + + true + DistributeFromPersWksp + + + true + EditActivatedOrders + + + true + EditBrandTemplates + + + true + EditCaseComments + + + true + EditEvent + + + true + EditHtmlTemplates + + + true + EditKnowledge + + + true + EditMyDashboards + + + true + EditMyReports + + + true + EditOppLineItemUnitPrice + + + true + EditPublicDocuments + + + true + EditPublicFilters + + + true + EditPublicTemplates + + + true + EditReadonlyFields + + + true + EditTask + + + true + EditTopics + + + true + EmailMass + + + true + EmailSingle + + + true + EnableCommunityAppLauncher + + + true + EnableNotifications + + + true + ExportReport + + + true + GiveRecognitionBadge + + + true + ImportCustomObjects + + + true + ImportLeads + + + true + ImportPersonal + + + true + InboundMigrationToolsUser + + + true + InstallPackaging + + + true + LightningConsoleAllowedForUser + + + true + LightningExperienceUser + + + true + ListEmailSend + + + true + ManageAnalyticSnapshots + + + true + ManageAuthProviders + + + true + ManageBusinessHourHolidays + + + true + ManageC360AConnections + + + true + ManageCMS + + + true + ManageCallCenters + + + true + ManageCases + + + true + ManageCategories + + + true + ManageCertificates + + + true + ManageContentPermissions + + + true + ManageContentProperties + + + true + ManageContentTypes + + + true + ManageCustomPermissions + + + true + ManageCustomReportTypes + + + true + ManageDashbdsInPubFolders + + + true + ManageDataCategories + + + true + ManageDataIntegrations + + + true + ManageDynamicDashboards + + + true + ManageEmailClientConfig + + + true + ManageEntitlements + + + true + ManageExchangeConfig + + + true + ManageHealthCheck + + + true + ManageHubConnections + + + true + ManageInteraction + + + true + ManageInternalUsers + + + true + ManageIpAddresses + + + true + ManageKnowledge + + + true + ManageKnowledgeImportExport + + + true + ManageLeads + + + true + ManageLoginAccessPolicies + + + true + ManageMobile + + + true + ManageNetworks + + + true + ManagePackageLicenses + + + true + ManagePasswordPolicies + + + true + ManageProfilesPermissionsets + + + true + ManagePropositions + + + true + ManagePvtRptsAndDashbds + + + true + ManageRecommendationStrategies + + + true + ManageReleaseUpdates + + + true + ManageRemoteAccess + + + true + ManageReportsInPubFolders + + + true + ManageRoles + + + true + ManageSearchPromotionRules + + + true + ManageSharing + + + true + ManageSolutions + + + true + ManageSubscriptions + + + true + ManageSynonyms + + + true + ManageTrustMeasures + + + true + ManageUnlistedGroups + + + true + ManageUsers + + + true + MassInlineEdit + + + true + MergeTopics + + + true + ModerateChatter + + + true + ModifyAllData + + + true + ModifyDataClassification + + + true + ModifyMetadata + + + true + NewReportBuilder + + + true + OutboundMigrationToolsUser + + + true + Packaging2 + + + true + Packaging2Delete + + + true + PrivacyDataAccess + + + true + RemoveDirectMessageMembers + + + true + ResetPasswords + + + true + RunReports + + + true + ScheduleJob + + + true + ScheduleReports + + + true + SelectFilesFromSalesforce + + + true + SendCustomNotifications + + + true + SendSitRequests + + + true + ShareInternalArticles + + + true + ShowCompanyNameAsUserBadge + + + true + SolutionImport + + + true + SubmitMacrosAllowed + + + true + SubscribeDashboardRolesGrps + + + true + SubscribeDashboardToOtherUsers + + + true + SubscribeReportRolesGrps + + + true + SubscribeReportToOtherUsers + + + true + SubscribeReportsRunAsUser + + + true + SubscribeToLightningDashboards + + + true + SubscribeToLightningReports + + + true + TransactionalEmailSend + + + true + TransferAnyCase + + + true + TransferAnyEntity + + + true + TransferAnyLead + + + true + UseTeamReassignWizards + + + true + UseWebLink + + + true + ViewAllData + + + true + ViewAllProfiles + + + true + ViewAllUsers + + + true + ViewDataAssessment + + + true + ViewDataCategories + + + true + ViewDeveloperName + + + true + ViewEventLogFiles + + + true + ViewFlowUsageAndFlowEventData + + + true + ViewHealthCheck + + + true + ViewHelpLink + + + true + ViewMyTeamsDashboards + + + true + ViewPublicDashboards + + + true + ViewPublicReports + + + true + ViewRoles + + + true + ViewSetup + + + true + ViewTrustMeasures + + + true + ViewUserPII + + diff --git a/package-lock.json b/package-lock.json index e86426c37..048973be6 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,9 +1,18 @@ { "name": "nebula-logger", - "version": "4.7.0", + "version": "4.7.1", "lockfileVersion": 1, "requires": true, "dependencies": { + "@ampproject/remapping": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/@ampproject/remapping/-/remapping-2.1.2.tgz", + "integrity": "sha512-hoyByceqwKirw7w3Z7gnIIZC3Wx3J484Y3L/cMpXFbr7d9ZQj2mODrirNzcJa+SM3UlpWXYvKV4RlRpFXlWgXg==", + "dev": true, + "requires": { + "@jridgewell/trace-mapping": "^0.3.0" + } + }, "@babel/code-frame": { "version": "7.16.7", "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.16.7.tgz", @@ -14,38 +23,38 @@ } }, "@babel/compat-data": { - "version": "7.16.4", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.16.4.tgz", - "integrity": "sha512-1o/jo7D+kC9ZjHX5v+EHrdjl3PhxMrLSOTGsOdHJ+KL8HCaEK6ehrVL2RS6oHDZp+L7xLirLrPmQtEng769J/Q==", + "version": "7.17.7", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.17.7.tgz", + "integrity": "sha512-p8pdE6j0a29TNGebNm7NzYZWB3xVZJBZ7XGs42uAKzQo8VQ3F0By/cQCtUEABwIqw5zo6WA4NbmxsfzADzMKnQ==", "dev": true }, "@babel/core": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.16.7.tgz", - "integrity": "sha512-aeLaqcqThRNZYmbMqtulsetOQZ/5gbR/dWruUCJcpas4Qoyy+QeagfDsPdMrqwsPRDNxJvBlRiZxxX7THO7qtA==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.17.9.tgz", + "integrity": "sha512-5ug+SfZCpDAkVp9SFIZAzlW18rlzsOcJGaetCjkySnrXXDUw9AR8cDUm1iByTmdWM6yxX6/zycaV76w3YTF2gw==", "dev": true, "requires": { + "@ampproject/remapping": "^2.1.0", "@babel/code-frame": "^7.16.7", - "@babel/generator": "^7.16.7", - "@babel/helper-compilation-targets": "^7.16.7", - "@babel/helper-module-transforms": "^7.16.7", - "@babel/helpers": "^7.16.7", - "@babel/parser": "^7.16.7", + "@babel/generator": "^7.17.9", + "@babel/helper-compilation-targets": "^7.17.7", + "@babel/helper-module-transforms": "^7.17.7", + "@babel/helpers": "^7.17.9", + "@babel/parser": "^7.17.9", "@babel/template": "^7.16.7", - "@babel/traverse": "^7.16.7", - "@babel/types": "^7.16.7", + "@babel/traverse": "^7.17.9", + "@babel/types": "^7.17.0", "convert-source-map": "^1.7.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", - "json5": "^2.1.2", - "semver": "^6.3.0", - "source-map": "^0.5.0" + "json5": "^2.2.1", + "semver": "^6.3.0" } }, "@babel/eslint-parser": { - "version": "7.16.5", - "resolved": "https://registry.npmjs.org/@babel/eslint-parser/-/eslint-parser-7.16.5.tgz", - "integrity": "sha512-mUqYa46lgWqHKQ33Q6LNCGp/wPR3eqOYTUixHFsfrSQqRxH0+WOzca75iEjFr5RDGH1dDz622LaHhLOzOuQRUA==", + "version": "7.17.0", + "resolved": "https://registry.npmjs.org/@babel/eslint-parser/-/eslint-parser-7.17.0.tgz", + "integrity": "sha512-PUEJ7ZBXbRkbq3qqM/jZ2nIuakUBqCYc7Qf52Lj7dlZ6zERnqisdHioL0l4wwQZnmskMeasqUNzLBFKs3nylXA==", "dev": true, "requires": { "eslint-scope": "^5.1.1", @@ -54,12 +63,12 @@ } }, "@babel/generator": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.16.7.tgz", - "integrity": "sha512-/ST3Sg8MLGY5HVYmrjOgL60ENux/HfO/CsUh7y4MalThufhE/Ff/6EibFDHi4jiDCaWfJKoqbE6oTh21c5hrRg==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.17.9.tgz", + "integrity": "sha512-rAdDousTwxbIxbz5I7GEQ3lUip+xVCXooZNbsydCWs3xA7ZsYOv+CFRdzGxRX78BmQHu9B1Eso59AOZQOJDEdQ==", "dev": true, "requires": { - "@babel/types": "^7.16.7", + "@babel/types": "^7.17.0", "jsesc": "^2.5.1", "source-map": "^0.5.0" } @@ -74,27 +83,27 @@ } }, "@babel/helper-compilation-targets": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.16.7.tgz", - "integrity": "sha512-mGojBwIWcwGD6rfqgRXVlVYmPAv7eOpIemUG3dGnDdCY4Pae70ROij3XmfrH6Fa1h1aiDylpglbZyktfzyo/hA==", + "version": "7.17.7", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.17.7.tgz", + "integrity": "sha512-UFzlz2jjd8kroj0hmCFV5zr+tQPi1dpC2cRsDV/3IEW8bJfCPrPpmcSN6ZS8RqIq4LXcmpipCQFPddyFA5Yc7w==", "dev": true, "requires": { - "@babel/compat-data": "^7.16.4", + "@babel/compat-data": "^7.17.7", "@babel/helper-validator-option": "^7.16.7", "browserslist": "^4.17.5", "semver": "^6.3.0" } }, "@babel/helper-create-class-features-plugin": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.16.7.tgz", - "integrity": "sha512-kIFozAvVfK05DM4EVQYKK+zteWvY85BFdGBRQBytRyY3y+6PX0DkDOn/CZ3lEuczCfrCxEzwt0YtP/87YPTWSw==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.17.9.tgz", + "integrity": "sha512-kUjip3gruz6AJKOq5i3nC6CoCEEF/oHH3cp6tOZhB+IyyyPyW0g1Gfsxn3mkk6S08pIA2y8GQh609v9G/5sHVQ==", "dev": true, "requires": { "@babel/helper-annotate-as-pure": "^7.16.7", "@babel/helper-environment-visitor": "^7.16.7", - "@babel/helper-function-name": "^7.16.7", - "@babel/helper-member-expression-to-functions": "^7.16.7", + "@babel/helper-function-name": "^7.17.9", + "@babel/helper-member-expression-to-functions": "^7.17.7", "@babel/helper-optimise-call-expression": "^7.16.7", "@babel/helper-replace-supers": "^7.16.7", "@babel/helper-split-export-declaration": "^7.16.7" @@ -110,23 +119,13 @@ } }, "@babel/helper-function-name": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.16.7.tgz", - "integrity": "sha512-QfDfEnIUyyBSR3HtrtGECuZ6DAyCkYFp7GHl75vFtTnn6pjKeK0T1DB5lLkFvBea8MdaiUABx3osbgLyInoejA==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.17.9.tgz", + "integrity": "sha512-7cRisGlVtiVqZ0MW0/yFB4atgpGLWEHUVYnb448hZK4x+vih0YO5UoS11XIYtZYqHd0dIPMdUSv8q5K4LdMnIg==", "dev": true, "requires": { - "@babel/helper-get-function-arity": "^7.16.7", "@babel/template": "^7.16.7", - "@babel/types": "^7.16.7" - } - }, - "@babel/helper-get-function-arity": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.16.7.tgz", - "integrity": "sha512-flc+RLSOBXzNzVhcLu6ujeHUrD6tANAOU5ojrRx/as+tbzf8+stUCj7+IfRRoAbEZqj/ahXEMsjhOhgeZsrnTw==", - "dev": true, - "requires": { - "@babel/types": "^7.16.7" + "@babel/types": "^7.17.0" } }, "@babel/helper-hoist-variables": { @@ -139,12 +138,12 @@ } }, "@babel/helper-member-expression-to-functions": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.16.7.tgz", - "integrity": "sha512-VtJ/65tYiU/6AbMTDwyoXGPKHgTsfRarivm+YbB5uAzKUyuPjgZSgAFeG87FCigc7KNHu2Pegh1XIT3lXjvz3Q==", + "version": "7.17.7", + "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.17.7.tgz", + "integrity": "sha512-thxXgnQ8qQ11W2wVUObIqDL4p148VMxkt5T/qpN5k2fboRyzFGFmKsTGViquyM5QHKUy48OZoca8kw4ajaDPyw==", "dev": true, "requires": { - "@babel/types": "^7.16.7" + "@babel/types": "^7.17.0" } }, "@babel/helper-module-imports": { @@ -157,19 +156,19 @@ } }, "@babel/helper-module-transforms": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.16.7.tgz", - "integrity": "sha512-gaqtLDxJEFCeQbYp9aLAefjhkKdjKcdh6DB7jniIGU3Pz52WAmP268zK0VgPz9hUNkMSYeH976K2/Y6yPadpng==", + "version": "7.17.7", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.17.7.tgz", + "integrity": "sha512-VmZD99F3gNTYB7fJRDTi+u6l/zxY0BE6OIxPSU7a50s6ZUQkHwSDmV92FfM+oCG0pZRVojGYhkR8I0OGeCVREw==", "dev": true, "requires": { "@babel/helper-environment-visitor": "^7.16.7", "@babel/helper-module-imports": "^7.16.7", - "@babel/helper-simple-access": "^7.16.7", + "@babel/helper-simple-access": "^7.17.7", "@babel/helper-split-export-declaration": "^7.16.7", "@babel/helper-validator-identifier": "^7.16.7", "@babel/template": "^7.16.7", - "@babel/traverse": "^7.16.7", - "@babel/types": "^7.16.7" + "@babel/traverse": "^7.17.3", + "@babel/types": "^7.17.0" } }, "@babel/helper-optimise-call-expression": { @@ -201,12 +200,12 @@ } }, "@babel/helper-simple-access": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.16.7.tgz", - "integrity": "sha512-ZIzHVyoeLMvXMN/vok/a4LWRy8G2v205mNP0XOuf9XRLyX5/u9CnVulUtDgUTama3lT+bf/UqucuZjqiGuTS1g==", + "version": "7.17.7", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.17.7.tgz", + "integrity": "sha512-txyMCGroZ96i+Pxr3Je3lzEJjqwaRC9buMUgtomcrLe5Nd0+fk1h0LLA+ixUF5OW7AhHuQ7Es1WcQJZmZsz2XA==", "dev": true, "requires": { - "@babel/types": "^7.16.7" + "@babel/types": "^7.17.0" } }, "@babel/helper-split-export-declaration": { @@ -231,20 +230,20 @@ "dev": true }, "@babel/helpers": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.16.7.tgz", - "integrity": "sha512-9ZDoqtfY7AuEOt3cxchfii6C7GDyyMBffktR5B2jvWv8u2+efwvpnVKXMWzNehqy68tKgAfSwfdw/lWpthS2bw==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.17.9.tgz", + "integrity": "sha512-cPCt915ShDWUEzEp3+UNRktO2n6v49l5RSnG9M5pS24hA+2FAc5si+Pn1i4VVbQQ+jh+bIZhPFQOJOzbrOYY1Q==", "dev": true, "requires": { "@babel/template": "^7.16.7", - "@babel/traverse": "^7.16.7", - "@babel/types": "^7.16.7" + "@babel/traverse": "^7.17.9", + "@babel/types": "^7.17.0" } }, "@babel/highlight": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.16.7.tgz", - "integrity": "sha512-aKpPMfLvGO3Q97V0qhw/V2SWNWlwfJknuwAunU7wZLSfrM4xTBvg7E5opUVi1kJTBKihE38CPg4nBiqX83PWYw==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.17.9.tgz", + "integrity": "sha512-J9PfEKCbFIv2X5bjTMiZu6Vf341N05QIY+d6FvVKynkG1S7G0j3I0QoRtWIrXhZ+/Nlb5Q0MzqL7TokEJ5BNHg==", "dev": true, "requires": { "@babel/helper-validator-identifier": "^7.16.7", @@ -253,9 +252,9 @@ } }, "@babel/parser": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.16.7.tgz", - "integrity": "sha512-sR4eaSrnM7BV7QPzGfEX5paG/6wrZM3I0HDzfIAK06ESvo9oy3xBuVBxE3MbQaKNhvg8g/ixjMWo2CGpzpHsDA==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.17.9.tgz", + "integrity": "sha512-vqUSBLP8dQHFPdPi9bc5GK9vRkYHJ49fsZdtoJ8EQ8ibpwk5rPKfvNIwChB0KVXcIjcepEBBd2VHC5r9Gy8ueg==", "dev": true }, "@babel/plugin-proposal-class-properties": { @@ -319,9 +318,9 @@ } }, "@babel/plugin-syntax-decorators": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-decorators/-/plugin-syntax-decorators-7.16.7.tgz", - "integrity": "sha512-vQ+PxL+srA7g6Rx6I1e15m55gftknl2X8GCUW1JTlkTaXZLJOS0UcaY0eK9jYT7IYf4awn6qwyghVHLDz1WyMw==", + "version": "7.17.0", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-decorators/-/plugin-syntax-decorators-7.17.0.tgz", + "integrity": "sha512-qWe85yCXsvDEluNP0OyeQjH63DlhAR3W7K9BxxU1MvbDb48tgBG+Ao6IJJ6smPDrrVzSQZrbF6donpkFBMcs3A==", "dev": true, "requires": { "@babel/helper-plugin-utils": "^7.16.7" @@ -427,14 +426,14 @@ } }, "@babel/plugin-transform-modules-commonjs": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.16.7.tgz", - "integrity": "sha512-h2RP2kE7He1ZWKyAlanMZrAbdv+Acw1pA8dQZhE025WJZE2z0xzFADAinXA9fxd5bn7JnM+SdOGcndGx1ARs9w==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.17.9.tgz", + "integrity": "sha512-2TBFd/r2I6VlYn0YRTz2JdazS+FoUuQ2rIFHoAxtyP/0G3D82SBLaRq9rnUkpqlLg03Byfl/+M32mpxjO6KaPw==", "dev": true, "requires": { - "@babel/helper-module-transforms": "^7.16.7", + "@babel/helper-module-transforms": "^7.17.7", "@babel/helper-plugin-utils": "^7.16.7", - "@babel/helper-simple-access": "^7.16.7", + "@babel/helper-simple-access": "^7.17.7", "babel-plugin-dynamic-import-node": "^2.3.3" } }, @@ -448,9 +447,9 @@ } }, "@babel/plugin-transform-typescript": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.16.7.tgz", - "integrity": "sha512-Hzx1lvBtOCWuCEwMmYOfpQpO7joFeXLgoPuzZZBtTxXqSqUGUubvFGZv2ygo1tB5Bp9q6PXV3H0E/kf7KM0RLA==", + "version": "7.16.8", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typescript/-/plugin-transform-typescript-7.16.8.tgz", + "integrity": "sha512-bHdQ9k7YpBDO2d0NVfkj51DpQcvwIzIusJ7mEUaMlbZq3Kt/U47j24inXZHQ5MDiYpCs+oZiwnXyKedE8+q7AQ==", "dev": true, "requires": { "@babel/helper-create-class-features-plugin": "^7.16.7", @@ -469,6 +468,25 @@ "@babel/plugin-transform-typescript": "^7.16.7" } }, + "@babel/runtime": { + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.17.9.tgz", + "integrity": "sha512-lSiBBvodq29uShpWGNbgFdKYNiFDo5/HIYsaCEY9ff4sb10x9jizo2+pRrSyF4jKZCXqgzuqBOQKbUm90gQwJg==", + "dev": true, + "requires": { + "regenerator-runtime": "^0.13.4" + } + }, + "@babel/runtime-corejs3": { + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/runtime-corejs3/-/runtime-corejs3-7.17.9.tgz", + "integrity": "sha512-WxYHHUWF2uZ7Hp1K+D1xQgbgkGUfA+5UPOegEXGt2Y5SMog/rYCVaifLZDbw8UkNXozEqqrZTy6bglL7xTaCOw==", + "dev": true, + "requires": { + "core-js-pure": "^3.20.2", + "regenerator-runtime": "^0.13.4" + } + }, "@babel/template": { "version": "7.16.7", "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.16.7.tgz", @@ -481,27 +499,27 @@ } }, "@babel/traverse": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.16.7.tgz", - "integrity": "sha512-8KWJPIb8c2VvY8AJrydh6+fVRo2ODx1wYBU2398xJVq0JomuLBZmVQzLPBblJgHIGYG4znCpUZUZ0Pt2vdmVYQ==", + "version": "7.17.9", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.17.9.tgz", + "integrity": "sha512-PQO8sDIJ8SIwipTPiR71kJQCKQYB5NGImbOviK8K+kg5xkNSYXLBupuX9QhatFowrsvo9Hj8WgArg3W7ijNAQw==", "dev": true, "requires": { "@babel/code-frame": "^7.16.7", - "@babel/generator": "^7.16.7", + "@babel/generator": "^7.17.9", "@babel/helper-environment-visitor": "^7.16.7", - "@babel/helper-function-name": "^7.16.7", + "@babel/helper-function-name": "^7.17.9", "@babel/helper-hoist-variables": "^7.16.7", "@babel/helper-split-export-declaration": "^7.16.7", - "@babel/parser": "^7.16.7", - "@babel/types": "^7.16.7", + "@babel/parser": "^7.17.9", + "@babel/types": "^7.17.0", "debug": "^4.1.0", "globals": "^11.1.0" } }, "@babel/types": { - "version": "7.16.7", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.16.7.tgz", - "integrity": "sha512-E8HuV7FO9qLpx6OtoGfUQ2cjIYnbFwvZWYBS+87EwtdMvmUPJSwykpovFB+8insbpF0uJcpr8KMUi64XZntZcg==", + "version": "7.17.0", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.17.0.tgz", + "integrity": "sha512-TmKSNO4D5rzhL5bjWFcVHHLETzfQ/AmbKpKPOSjlP0WoHZ6L911fgoOKY4Alp/emzG4cHJdyN49zpgkbXFEHHw==", "dev": true, "requires": { "@babel/helper-validator-identifier": "^7.16.7", @@ -514,10 +532,17 @@ "integrity": "sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==", "dev": true }, + "@colors/colors": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@colors/colors/-/colors-1.5.0.tgz", + "integrity": "sha512-ooWCrlZP11i8GImSjTHYHLkvFDP48nS4+204nGb1RiX/WXYHmJA2III9/e2DWVabCESdW7hBAEzHRqUn9OUVvQ==", + "dev": true, + "optional": true + }, "@cparra/apexdocs": { - "version": "1.14.0", - "resolved": "https://registry.npmjs.org/@cparra/apexdocs/-/apexdocs-1.14.0.tgz", - "integrity": "sha512-2EruYNdNado6MkgGGxrjD9cve+c6GN/3g/+wNtEuMys6K6snOUIWym8hhkDiosmEGZUMizd9rWg49KfgzYIjXw==", + "version": "1.13.7", + "resolved": "https://registry.npmjs.org/@cparra/apexdocs/-/apexdocs-1.13.7.tgz", + "integrity": "sha512-XqMQM6mEcJY7vRv50TrR1PRWGNMxNRCe7i1JKe8rN1B4itNuT0aspGbia9FlJQA4XgfLvOW6y4htErg3P1mGew==", "dev": true, "requires": { "html-entities": "^2.3.2", @@ -526,16 +551,16 @@ } }, "@eslint/eslintrc": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-1.0.5.tgz", - "integrity": "sha512-BLxsnmK3KyPunz5wmCCpqy0YelEoxxGmH73Is+Z74oOTMtExcjkr3dDR6quwrjh1YspA8DH9gnX1o069KiS9AQ==", + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@eslint/eslintrc/-/eslintrc-1.2.1.tgz", + "integrity": "sha512-bxvbYnBPN1Gibwyp6NrpnFzA3YtRL3BBAyEAFVIpNTm2Rn4Vy87GA5M4aSn3InRrlsbX5N0GW7XIx+U4SAEKdQ==", "dev": true, "requires": { "ajv": "^6.12.4", "debug": "^4.3.2", - "espree": "^9.2.0", + "espree": "^9.3.1", "globals": "^13.9.0", - "ignore": "^4.0.6", + "ignore": "^5.2.0", "import-fresh": "^3.2.1", "js-yaml": "^4.1.0", "minimatch": "^3.0.4", @@ -555,37 +580,31 @@ "dev": true }, "eslint-visitor-keys": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.2.0.tgz", - "integrity": "sha512-IOzT0X126zn7ALX0dwFiUQEdsfzrm4+ISsQS8nukaJXwEyYKRSnEIIDULYg1mCtGp7UUXgfGl7BIolXREQK+XQ==", + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.3.0.tgz", + "integrity": "sha512-mQ+suqKJVyeuwGYHAdjMFqjCyfl8+Ldnxuyp3ldiMBFKkvytrXUZWaiPCEav8qDHKty44bD+qV1IP4T+w+xXRA==", "dev": true }, "espree": { - "version": "9.3.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-9.3.0.tgz", - "integrity": "sha512-d/5nCsb0JcqsSEeQzFZ8DH1RmxPcglRWh24EFTlUEmCKoehXGdpsx0RkHDubqUI8LSAIKMQp4r9SzQ3n+sm4HQ==", + "version": "9.3.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.3.1.tgz", + "integrity": "sha512-bvdyLmJMfwkV3NCRl5ZhJf22zBFo1y8bYh3VYb+bfzqNB4Je68P2sSuXyuFquzWLebHpNd2/d5uv7yoP9ISnGQ==", "dev": true, "requires": { "acorn": "^8.7.0", "acorn-jsx": "^5.3.1", - "eslint-visitor-keys": "^3.1.0" + "eslint-visitor-keys": "^3.3.0" } }, "globals": { - "version": "13.12.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-13.12.0.tgz", - "integrity": "sha512-uS8X6lSKN2JumVoXrbUz+uG4BYG+eiawqm3qFcT7ammfbUHeCBoJMlHcec/S3krSk73/AE/f0szYFmgAA3kYZg==", + "version": "13.13.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.13.0.tgz", + "integrity": "sha512-EQ7Q18AJlPwp3vUDL4mKA0KXrXyNIQyWon6T6XQiBQF0XHvRsiCSrWmmeATpUzdJN2HhWZU6Pdl0a9zdep5p6A==", "dev": true, "requires": { "type-fest": "^0.20.2" } }, - "ignore": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", - "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", - "dev": true - }, "js-yaml": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", @@ -618,10 +637,46 @@ "@hapi/hoek": "^9.0.0" } }, + "@heroku-cli/color": { + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/@heroku-cli/color/-/color-1.1.14.tgz", + "integrity": "sha512-2JYy//YE2YINTe21hpdVMBNc7aYFkgDeY9JUz/BCjFZmYLn0UjGaCc4BpTcMGXNJwuqoUenw2WGOFGHsJqlIDw==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "chalk": "^2.4.1", + "strip-ansi": "^5.0.0", + "supports-color": "^5.5.0", + "tslib": "^1.9.3" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, "@humanwhocodes/config-array": { - "version": "0.9.3", - "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.9.3.tgz", - "integrity": "sha512-3xSMlXHh03hCcCmFc0rbKp3Ivt2PFEJnQUJDDMTJQ2wkECZWdq4GePs2ctc5H8zV+cHPaq8k2vU8mrQjA6iHdQ==", + "version": "0.9.5", + "resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.9.5.tgz", + "integrity": "sha512-ObyMyWxZiCu/yTisA7uzx81s40xR2fD5Cg/2Kq7G02ajkNubJf6BopgDTmDyc3U7sXpNKM8cYOw7s7Tyr+DnCw==", "dev": true, "requires": { "@humanwhocodes/object-schema": "^1.2.1", @@ -693,17 +748,48 @@ "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", "dev": true }, + "@jayree/sfdx-plugin-prettier": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@jayree/sfdx-plugin-prettier/-/sfdx-plugin-prettier-1.1.4.tgz", + "integrity": "sha512-cJflGJEpuxPo1vrDBahKt2EFS1WGnmdYm7djVxG3rpT7X6FHdNoZt6X3ptCCfpPwUX9oyTqj+WDB4CleyhguBA==", + "dev": true, + "requires": { + "@oclif/config": "^1.18.3", + "@oclif/errors": "^1.3.5", + "@prettier/plugin-xml": "^1.2.0", + "@salesforce/core": "^2.35.3", + "@salesforce/kit": "^1.5.33", + "cli-progress": "^3.10.0", + "debug": "^4.3.3", + "fs-extra": "^10.0.1", + "prettier": "^2.5.1", + "prettier-plugin-apex": "^1.10.0", + "tslib": "^2.3.1" + }, + "dependencies": { + "@prettier/plugin-xml": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@prettier/plugin-xml/-/plugin-xml-1.2.0.tgz", + "integrity": "sha512-bFvVAZKs59XNmntYjyefn3K4TBykS6E+d6ZW8IcylAs88ZO+TzLhp0dPpi0VKfPzq1Nb+kpDnPRTiwb4zY6NgA==", + "dev": true, + "requires": { + "@xml-tools/parser": "^1.0.11", + "prettier": ">=2.3" + } + } + } + }, "@jest/console": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/@jest/console/-/console-27.4.2.tgz", - "integrity": "sha512-xknHThRsPB/To1FUbi6pCe43y58qFC03zfb6R7fDb/FfC7k2R3i1l+izRBJf8DI46KhYGRaF14Eo9A3qbBoixg==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/console/-/console-27.5.1.tgz", + "integrity": "sha512-kZ/tNpS3NXn0mlXXXPNuDZnb4c0oZ20r4K5eemM2k30ZC3G0T02nXUvyhf5YdbXWHPEJLc9qGLxEZ216MdL+Zg==", "dev": true, "requires": { - "@jest/types": "^27.4.2", + "@jest/types": "^27.5.1", "@types/node": "*", "chalk": "^4.0.0", - "jest-message-util": "^27.4.2", - "jest-util": "^27.4.2", + "jest-message-util": "^27.5.1", + "jest-util": "^27.5.1", "slash": "^3.0.0" }, "dependencies": { @@ -759,41 +845,50 @@ } }, "@jest/core": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/@jest/core/-/core-27.4.5.tgz", - "integrity": "sha512-3tm/Pevmi8bDsgvo73nX8p/WPng6KWlCyScW10FPEoN1HU4pwI83tJ3TsFvi1FfzsjwUlMNEPowgb/rPau/LTQ==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/core/-/core-27.5.1.tgz", + "integrity": "sha512-AK6/UTrvQD0Cd24NSqmIA6rKsu0tKIxfiCducZvqxYdmMisOYAsdItspT+fQDQYARPf8XgjAFZi0ogW2agH5nQ==", "dev": true, "requires": { - "@jest/console": "^27.4.2", - "@jest/reporters": "^27.4.5", - "@jest/test-result": "^27.4.2", - "@jest/transform": "^27.4.5", - "@jest/types": "^27.4.2", + "@jest/console": "^27.5.1", + "@jest/reporters": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", "@types/node": "*", "ansi-escapes": "^4.2.1", "chalk": "^4.0.0", "emittery": "^0.8.1", "exit": "^0.1.2", - "graceful-fs": "^4.2.4", - "jest-changed-files": "^27.4.2", - "jest-config": "^27.4.5", - "jest-haste-map": "^27.4.5", - "jest-message-util": "^27.4.2", - "jest-regex-util": "^27.4.0", - "jest-resolve": "^27.4.5", - "jest-resolve-dependencies": "^27.4.5", - "jest-runner": "^27.4.5", - "jest-runtime": "^27.4.5", - "jest-snapshot": "^27.4.5", - "jest-util": "^27.4.2", - "jest-validate": "^27.4.2", - "jest-watcher": "^27.4.2", + "graceful-fs": "^4.2.9", + "jest-changed-files": "^27.5.1", + "jest-config": "^27.5.1", + "jest-haste-map": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-resolve-dependencies": "^27.5.1", + "jest-runner": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "jest-watcher": "^27.5.1", "micromatch": "^4.0.4", "rimraf": "^3.0.0", "slash": "^3.0.0", "strip-ansi": "^6.0.0" }, "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, "ansi-styles": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", @@ -834,6 +929,15 @@ "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true }, + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, "supports-color": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", @@ -846,68 +950,68 @@ } }, "@jest/environment": { - "version": "27.4.4", - "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-27.4.4.tgz", - "integrity": "sha512-q+niMx7cJgt/t/b6dzLOh4W8Ef/8VyKG7hxASK39jakijJzbFBGpptx3RXz13FFV7OishQ9lTbv+dQ5K3EhfDQ==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/environment/-/environment-27.5.1.tgz", + "integrity": "sha512-/WQjhPJe3/ghaol/4Bq480JKXV/Rfw8nQdN7f41fM8VDHLcxKXou6QyXAh3EFr9/bVG3x74z1NWDkP87EiY8gA==", "dev": true, "requires": { - "@jest/fake-timers": "^27.4.2", - "@jest/types": "^27.4.2", + "@jest/fake-timers": "^27.5.1", + "@jest/types": "^27.5.1", "@types/node": "*", - "jest-mock": "^27.4.2" + "jest-mock": "^27.5.1" } }, "@jest/fake-timers": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-27.4.2.tgz", - "integrity": "sha512-f/Xpzn5YQk5adtqBgvw1V6bF8Nx3hY0OIRRpCvWcfPl0EAjdqWPdhH3t/3XpiWZqtjIEHDyMKP9ajpva1l4Zmg==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/fake-timers/-/fake-timers-27.5.1.tgz", + "integrity": "sha512-/aPowoolwa07k7/oM3aASneNeBGCmGQsc3ugN4u6s4C/+s5M64MFo/+djTdiwcbQlRfFElGuDXWzaWj6QgKObQ==", "dev": true, "requires": { - "@jest/types": "^27.4.2", + "@jest/types": "^27.5.1", "@sinonjs/fake-timers": "^8.0.1", "@types/node": "*", - "jest-message-util": "^27.4.2", - "jest-mock": "^27.4.2", - "jest-util": "^27.4.2" + "jest-message-util": "^27.5.1", + "jest-mock": "^27.5.1", + "jest-util": "^27.5.1" } }, "@jest/globals": { - "version": "27.4.4", - "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-27.4.4.tgz", - "integrity": "sha512-bqpqQhW30BOreXM8bA8t8JbOQzsq/WnPTnBl+It3UxAD9J8yxEAaBEylHx1dtBapAr/UBk8GidXbzmqnee8tYQ==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/globals/-/globals-27.5.1.tgz", + "integrity": "sha512-ZEJNB41OBQQgGzgyInAv0UUfDDj3upmHydjieSxFvTRuZElrx7tXg/uVQ5hYVEwiXs3+aMsAeEc9X7xiSKCm4Q==", "dev": true, "requires": { - "@jest/environment": "^27.4.4", - "@jest/types": "^27.4.2", - "expect": "^27.4.2" + "@jest/environment": "^27.5.1", + "@jest/types": "^27.5.1", + "expect": "^27.5.1" } }, "@jest/reporters": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-27.4.5.tgz", - "integrity": "sha512-3orsG4vi8zXuBqEoy2LbnC1kuvkg1KQUgqNxmxpQgIOQEPeV0onvZu+qDQnEoX8qTQErtqn/xzcnbpeTuOLSiA==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/reporters/-/reporters-27.5.1.tgz", + "integrity": "sha512-cPXh9hWIlVJMQkVk84aIvXuBB4uQQmFqZiacloFuGiP3ah1sbCxCosidXFDfqG8+6fO1oR2dTJTlsOy4VFmUfw==", "dev": true, "requires": { "@bcoe/v8-coverage": "^0.2.3", - "@jest/console": "^27.4.2", - "@jest/test-result": "^27.4.2", - "@jest/transform": "^27.4.5", - "@jest/types": "^27.4.2", + "@jest/console": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", "@types/node": "*", "chalk": "^4.0.0", "collect-v8-coverage": "^1.0.0", "exit": "^0.1.2", "glob": "^7.1.2", - "graceful-fs": "^4.2.4", + "graceful-fs": "^4.2.9", "istanbul-lib-coverage": "^3.0.0", - "istanbul-lib-instrument": "^4.0.3", + "istanbul-lib-instrument": "^5.1.0", "istanbul-lib-report": "^3.0.0", "istanbul-lib-source-maps": "^4.0.0", - "istanbul-reports": "^3.0.2", - "jest-haste-map": "^27.4.5", - "jest-resolve": "^27.4.5", - "jest-util": "^27.4.2", - "jest-worker": "^27.4.5", + "istanbul-reports": "^3.1.3", + "jest-haste-map": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", "slash": "^3.0.0", "source-map": "^0.6.0", "string-length": "^4.0.1", @@ -973,13 +1077,13 @@ } }, "@jest/source-map": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-27.4.0.tgz", - "integrity": "sha512-Ntjx9jzP26Bvhbm93z/AKcPRj/9wrkI88/gK60glXDx1q+IeI0rf7Lw2c89Ch6ofonB0On/iRDreQuQ6te9pgQ==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/source-map/-/source-map-27.5.1.tgz", + "integrity": "sha512-y9NIHUYF3PJRlHk98NdC/N1gl88BL08aQQgu4k4ZopQkCw9t9cV8mtl3TV8b/YCB8XaVTFrmUTAJvjsntDireg==", "dev": true, "requires": { "callsites": "^3.0.0", - "graceful-fs": "^4.2.4", + "graceful-fs": "^4.2.9", "source-map": "^0.6.0" }, "dependencies": { @@ -992,47 +1096,47 @@ } }, "@jest/test-result": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-27.4.2.tgz", - "integrity": "sha512-kr+bCrra9jfTgxHXHa2UwoQjxvQk3Am6QbpAiJ5x/50LW8llOYrxILkqY0lZRW/hu8FXesnudbql263+EW9iNA==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-27.5.1.tgz", + "integrity": "sha512-EW35l2RYFUcUQxFJz5Cv5MTOxlJIQs4I7gxzi2zVU7PJhOwfYq1MdC5nhSmYjX1gmMmLPvB3sIaC+BkcHRBfag==", "dev": true, "requires": { - "@jest/console": "^27.4.2", - "@jest/types": "^27.4.2", + "@jest/console": "^27.5.1", + "@jest/types": "^27.5.1", "@types/istanbul-lib-coverage": "^2.0.0", "collect-v8-coverage": "^1.0.0" } }, "@jest/test-sequencer": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-27.4.5.tgz", - "integrity": "sha512-n5woIn/1v+FT+9hniymHPARA9upYUmfi5Pw9ewVwXCDlK4F5/Gkees9v8vdjGdAIJ2MPHLHodiajLpZZanWzEQ==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/test-sequencer/-/test-sequencer-27.5.1.tgz", + "integrity": "sha512-LCheJF7WB2+9JuCS7VB/EmGIdQuhtqjRNI9A43idHv3E4KltCTsPsLxvdaubFHSYwY/fNjMWjl6vNRhDiN7vpQ==", "dev": true, "requires": { - "@jest/test-result": "^27.4.2", - "graceful-fs": "^4.2.4", - "jest-haste-map": "^27.4.5", - "jest-runtime": "^27.4.5" + "@jest/test-result": "^27.5.1", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-runtime": "^27.5.1" } }, "@jest/transform": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-27.4.5.tgz", - "integrity": "sha512-PuMet2UlZtlGzwc6L+aZmR3I7CEBpqadO03pU40l2RNY2fFJ191b9/ITB44LNOhVtsyykx0OZvj0PCyuLm7Eew==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/transform/-/transform-27.5.1.tgz", + "integrity": "sha512-ipON6WtYgl/1329g5AIJVbUuEh0wZVbdpGwC99Jw4LwuoBNS95MVphU6zOeD9pDkon+LLbFL7lOQRapbB8SCHw==", "dev": true, "requires": { "@babel/core": "^7.1.0", - "@jest/types": "^27.4.2", - "babel-plugin-istanbul": "^6.0.0", + "@jest/types": "^27.5.1", + "babel-plugin-istanbul": "^6.1.1", "chalk": "^4.0.0", "convert-source-map": "^1.4.0", "fast-json-stable-stringify": "^2.0.0", - "graceful-fs": "^4.2.4", - "jest-haste-map": "^27.4.5", - "jest-regex-util": "^27.4.0", - "jest-util": "^27.4.2", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-util": "^27.5.1", "micromatch": "^4.0.4", - "pirates": "^4.0.1", + "pirates": "^4.0.4", "slash": "^3.0.0", "source-map": "^0.6.1", "write-file-atomic": "^3.0.0" @@ -1096,9 +1200,9 @@ } }, "@jest/types": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.4.2.tgz", - "integrity": "sha512-j35yw0PMTPpZsUoOBiuHzr1zTYoad1cVIE0ajEjcrJONxxrko/IRGKkXx3os0Nsi4Hu3+5VmDbVfq5WhG/pWAg==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/@jest/types/-/types-27.5.1.tgz", + "integrity": "sha512-Cx46iJ9QpwQTjIdq5VJu2QTMMs3QlEjI0x1QbBP5W1+nMzyc2XmimiRR/CbX9TO0cPTeUlxWMOu8mslYsJ8DEw==", "dev": true, "requires": { "@types/istanbul-lib-coverage": "^2.0.0", @@ -1160,9 +1264,9 @@ } }, "@jongpie/sfdx-bummer-plugin": { - "version": "0.0.17", - "resolved": "https://registry.npmjs.org/@jongpie/sfdx-bummer-plugin/-/sfdx-bummer-plugin-0.0.17.tgz", - "integrity": "sha512-payglGCjzRFyUbl8LmSwI2x+FoD6cxBWj3G8l575EOJDBi5CwzSpkjihO+WdL4nkkfVXemD2hFEKnsvwMDTLKA==", + "version": "0.0.18", + "resolved": "https://registry.npmjs.org/@jongpie/sfdx-bummer-plugin/-/sfdx-bummer-plugin-0.0.18.tgz", + "integrity": "sha512-vJIwBb0adf/BfBvqxoC9YDXvQ/avRrzWnqf52Xu141V32gMB9zJg69/jCiSPT1To4d0amLtJ7u0zA/rPoI+/FA==", "dev": true, "requires": { "@oclif/command": "^1", @@ -1172,103 +1276,34 @@ "@salesforce/core": "^2.35.0", "fs-extra": "^10.0.0", "tslib": "^2.3" - }, - "dependencies": { - "@salesforce/command": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-4.2.2.tgz", - "integrity": "sha512-2QEtPIMaeRyUEnLmPHJ1PhfKDJBupfQS5T4nG8rXpK2yOznBu48aPWaWCYErrxyC0bPa5eoFACeyPWz1k9QMog==", - "dev": true, - "requires": { - "@oclif/command": "^1.8.1", - "@oclif/errors": "^1.2.2", - "@oclif/parser": "3.8.6", - "@oclif/plugin-help": "^2.2.0", - "@oclif/test": "^1.2.4", - "@salesforce/core": "^2.35.0", - "@salesforce/kit": "^1.5.17", - "@salesforce/ts-types": "^1.5.20", - "chalk": "^2.4.2", - "cli-ux": "^4.9.3" - } - }, - "@salesforce/core": { - "version": "2.35.2", - "resolved": "https://registry.npmjs.org/@salesforce/core/-/core-2.35.2.tgz", - "integrity": "sha512-/f+nZFJf7wRVqZGxnCFGOICCLFJkx37oC/c1M8ICCPH0YRNa0GvlQQv7H/WYtej4OzA0HfnHkhxwCUuCQj5EHw==", - "dev": true, - "requires": { - "@salesforce/bunyan": "^2.0.0", - "@salesforce/kit": "^1.5.17", - "@salesforce/schemas": "^1.0.1", - "@salesforce/ts-types": "^1.5.20", - "@types/graceful-fs": "^4.1.5", - "@types/jsforce": "^1.9.38", - "@types/mkdirp": "^1.0.1", - "archiver": "^5.3.0", - "debug": "^3.1.0", - "faye": "^1.4.0", - "graceful-fs": "^4.2.4", - "js2xmlparser": "^4.0.1", - "jsen": "0.6.6", - "jsforce": "^1.11.0", - "jsonwebtoken": "8.5.0", - "mkdirp": "1.0.4", - "semver": "^7.3.5", - "ts-retry-promise": "^0.6.0" - } - }, - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "fs-extra": { - "version": "10.0.1", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.0.1.tgz", - "integrity": "sha512-NbdoVMZso2Lsrn/QwLXOy6rm0ufY2zEOKCDzJR/0kBsb0E6qed0P3iYK+Ath3BfvXEeu4JhEtXLgILx5psUfag==", - "dev": true, - "requires": { - "graceful-fs": "^4.2.0", - "jsonfile": "^6.0.1", - "universalify": "^2.0.0" - } - }, - "jsonfile": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", - "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.6", - "universalify": "^2.0.0" - } - }, - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", - "dev": true, - "requires": { - "lru-cache": "^6.0.0" - } - }, - "universalify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", - "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", - "dev": true - } + } + }, + "@jridgewell/resolve-uri": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.0.5.tgz", + "integrity": "sha512-VPeQ7+wH0itvQxnG+lIzWgkysKIr3L9sslimFW55rHMdGu/qCQ5z5h9zq4gI8uBtqkpHhsF4Z/OwExufUCThew==", + "dev": true + }, + "@jridgewell/sourcemap-codec": { + "version": "1.4.11", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.11.tgz", + "integrity": "sha512-Fg32GrJo61m+VqYSdRSjRXMjQ06j8YIYfcTqndLYVAaHmroZHLJZCydsWBOTDqXS2v+mjxohBWEMfg97GXmYQg==", + "dev": true + }, + "@jridgewell/trace-mapping": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.4.tgz", + "integrity": "sha512-vFv9ttIedivx0ux3QSjhgtCVjPZd5l46ZOMDSCwnH1yUO2e964gO8LZGyv2QkqcgR6TnBU1v+1IFqmeoG+0UJQ==", + "dev": true, + "requires": { + "@jridgewell/resolve-uri": "^3.0.3", + "@jridgewell/sourcemap-codec": "^1.4.10" } }, "@ljharb/eslint-config": { - "version": "20.2.3", - "resolved": "https://registry.npmjs.org/@ljharb/eslint-config/-/eslint-config-20.2.3.tgz", - "integrity": "sha512-Rk3IhGn8N7++t/1EQA2Op+0KvYZBBMmCyC8ICxA1gpDLnof64UVdoeUrn1h9YWNCnUMopm/xwH0GFirnWfD9lw==", + "version": "21.0.0", + "resolved": "https://registry.npmjs.org/@ljharb/eslint-config/-/eslint-config-21.0.0.tgz", + "integrity": "sha512-gk0AWYdJVA3UumN02ZjtGl1Q1+igH/EqhPgZMqJLFjhrSN54JzHu7ggVps7hdVnCKxwNc11myqrGrEYrN4Rq8A==", "dev": true, "requires": { "eslint-scope": "=7.1.0" @@ -1368,12 +1403,32 @@ "dev": true }, "@lwc/eslint-plugin-lwc": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/@lwc/eslint-plugin-lwc/-/eslint-plugin-lwc-1.1.1.tgz", - "integrity": "sha512-e+66ZydcQ0HFs/hIsq2JXMWVDDG2BoYWCG2ZBs/Y23bLFAE+iPFOmli5cHS9QqVOYzx9HlOTkt0h/XweNoQ5xg==", + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@lwc/eslint-plugin-lwc/-/eslint-plugin-lwc-1.2.1.tgz", + "integrity": "sha512-13a5XOultegAc4i4SyFk9367KOvvRy40zbRSKkApcNa9uMI6n9/C/yFMvwve6Mp6PaIsGXlKFWNkzwj+nADEzw==", "dev": true, "requires": { - "minimatch": "^3.0.4" + "minimatch": "^5.0.1" + }, + "dependencies": { + "brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0" + } + }, + "minimatch": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.0.1.tgz", + "integrity": "sha512-nLDxIFRyhDblz3qMuq+SoRZED4+miJ/G+tdDrjkkkRnjAsBexeGpgjLEQ0blJy7rHhR2b93rhQY4SvyWu9v03g==", + "dev": true, + "requires": { + "brace-expansion": "^2.0.1" + } + } } }, "@lwc/jest-preset": { @@ -1472,11 +1527,21 @@ "dev": true }, "@mdn/browser-compat-data": { - "version": "3.3.14", - "resolved": "https://registry.npmjs.org/@mdn/browser-compat-data/-/browser-compat-data-3.3.14.tgz", - "integrity": "sha512-n2RC9d6XatVbWFdHLimzzUJxJ1KY8LdjqrW6YvGPiRmsHkhOUx74/Ct10x5Yo7bC/Jvqx7cDEW8IMPv/+vwEzA==", + "version": "4.1.16", + "resolved": "https://registry.npmjs.org/@mdn/browser-compat-data/-/browser-compat-data-4.1.16.tgz", + "integrity": "sha512-bqXpkAfQgSD1jUNnn+kKtaHDDMPFz2kij4C71euV4fBv+YCmzkB8TKIoSuUHEdjkE0s55WoQ2OZk/ullmbiUOA==", "dev": true }, + "@mrmlnc/readdir-enhanced": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz", + "integrity": "sha512-bPHp6Ji8b41szTOcaP63VlnbbO5Ny6dwAATtY6JTjh5N2OLrb5Qk/Th5cRkRQhkWCt+EJsYrNB0MiL+Gpn6e3g==", + "dev": true, + "requires": { + "call-me-maybe": "^1.0.1", + "glob-to-regexp": "^0.3.0" + } + }, "@nodelib/fs.scandir": { "version": "2.1.5", "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", @@ -1503,43 +1568,302 @@ "fastq": "^1.6.0" } }, - "@oclif/command": { - "version": "1.8.15", - "resolved": "https://registry.npmjs.org/@oclif/command/-/command-1.8.15.tgz", - "integrity": "sha512-lMRsr38sm4XBDBt2u88nvGyk3YxMjQenfX1HM140ZckTVqwuvFVZLP4HZ0rSAK9FdJzfryO/tH5hovdiC6bgsw==", + "@oclif/color": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/@oclif/color/-/color-0.1.2.tgz", + "integrity": "sha512-M9o+DOrb8l603qvgz1FogJBUGLqcMFL1aFg2ZEL0FbXJofiNTLOWIeB4faeZTLwE6dt0xH9GpCVpzksMMzGbmA==", "dev": true, "requires": { - "@oclif/config": "^1.18.2", - "@oclif/errors": "^1.3.5", - "@oclif/help": "^1.0.1", - "@oclif/parser": "^3.8.6", - "debug": "^4.1.1", - "semver": "^7.3.2" + "ansi-styles": "^3.2.1", + "chalk": "^3.0.0", + "strip-ansi": "^5.2.0", + "supports-color": "^5.4.0", + "tslib": "^1" }, "dependencies": { - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true + }, + "chalk": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-3.0.0.tgz", + "integrity": "sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg==", "dev": true, "requires": { - "lru-cache": "^6.0.0" + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } } - } - } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "@oclif/command": { + "version": "1.8.16", + "resolved": "https://registry.npmjs.org/@oclif/command/-/command-1.8.16.tgz", + "integrity": "sha512-rmVKYEsKzurfRU0xJz+iHelbi1LGlihIWZ7Qvmb/CBz1EkhL7nOkW4SVXmG2dA5Ce0si2gr88i6q4eBOMRNJ1w==", + "dev": true, + "requires": { + "@oclif/config": "^1.18.2", + "@oclif/errors": "^1.3.5", + "@oclif/help": "^1.0.1", + "@oclif/parser": "^3.8.6", + "debug": "^4.1.1", + "semver": "^7.3.2" + }, + "dependencies": { + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } + } }, "@oclif/config": { - "version": "1.18.2", - "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.2.tgz", - "integrity": "sha512-cE3qfHWv8hGRCP31j7fIS7BfCflm/BNZ2HNqHexH+fDrdF2f1D5S8VmXWLC77ffv3oDvWyvE9AZeR0RfmHCCaA==", + "version": "1.18.3", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.3.tgz", + "integrity": "sha512-sBpko86IrTscc39EvHUhL+c++81BVTsIZ3ETu/vG+cCdi0N6vb2DoahR67A9FI2CGnxRRHjnTfa3m6LulwNATA==", "dev": true, "requires": { - "@oclif/errors": "^1.3.3", + "@oclif/errors": "^1.3.5", "@oclif/parser": "^3.8.0", "debug": "^4.1.1", "globby": "^11.0.1", "is-wsl": "^2.1.1", - "tslib": "^2.0.0" + "tslib": "^2.3.1" + } + }, + "@oclif/core": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/@oclif/core/-/core-1.7.0.tgz", + "integrity": "sha512-I4q4qgtnNG7ef4sBDrJhwADdi7RExQV7LnflnXaWZZDiUoqF9AdOjC4jjx40MK0rRrCehCUtPNZBcr3WmxuS4Q==", + "dev": true, + "requires": { + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^3.0.2", + "ansi-escapes": "^4.3.2", + "ansi-styles": "^4.3.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.2", + "clean-stack": "^3.0.1", + "cli-progress": "^3.10.0", + "debug": "^4.3.3", + "ejs": "^3.1.6", + "fs-extra": "^9.1.0", + "get-package-type": "^0.1.0", + "globby": "^11.1.0", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.14.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.3", + "object-treeify": "^1.1.33", + "password-prompt": "^1.1.2", + "semver": "^7.3.5", + "string-width": "^4.2.3", + "strip-ansi": "^6.0.1", + "supports-color": "^8.1.1", + "supports-hyperlinks": "^2.2.0", + "tslib": "^2.3.1", + "widest-line": "^3.1.0", + "wrap-ansi": "^7.0.0" + }, + "dependencies": { + "@oclif/screen": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@oclif/screen/-/screen-3.0.2.tgz", + "integrity": "sha512-S/SF/XYJeevwIgHFmVDAFRUvM3m+OjhvCAYMk78ZJQCYCQ5wS7j+LTt1ZEv2jpEEGg2tx/F6TYYWxddNAYHrFQ==", + "dev": true + }, + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "requires": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6", + "universalify": "^2.0.0" + } + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true + } } }, "@oclif/errors": { @@ -1553,6 +1877,31 @@ "indent-string": "^4.0.0", "strip-ansi": "^6.0.0", "wrap-ansi": "^7.0.0" + }, + "dependencies": { + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + } + } + }, + "@oclif/fixpack": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/@oclif/fixpack/-/fixpack-2.3.1.tgz", + "integrity": "sha512-ynDtqYke7WBjyAK5KQiOoNm2XGm4wBfNDzjbpq3k4NEarA5wRW0PMjdP2FkTAJE2NI3tA6I7DXxxqBzwyt+fBg==", + "dev": true, + "requires": { + "alce": "1.2.0", + "colors": "1.4.0", + "extend-object": "^1.0.0", + "rc": "^1.2.8" } }, "@oclif/help": { @@ -1572,6 +1921,20 @@ "wrap-ansi": "^6.2.0" }, "dependencies": { + "@oclif/config": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.2.tgz", + "integrity": "sha512-cE3qfHWv8hGRCP31j7fIS7BfCflm/BNZ2HNqHexH+fDrdF2f1D5S8VmXWLC77ffv3oDvWyvE9AZeR0RfmHCCaA==", + "dev": true, + "requires": { + "@oclif/errors": "^1.3.3", + "@oclif/parser": "^3.8.0", + "debug": "^4.1.1", + "globby": "^11.0.1", + "is-wsl": "^2.1.1", + "tslib": "^2.0.0" + } + }, "ansi-styles": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", @@ -1641,15 +2004,15 @@ "dev": true }, "@oclif/parser": { - "version": "3.8.6", - "resolved": "https://registry.npmjs.org/@oclif/parser/-/parser-3.8.6.tgz", - "integrity": "sha512-tXb0NKgSgNxmf6baN6naK+CCwOueaFk93FG9u202U7mTBHUKsioOUlw1SG/iPi9aJM3WE4pHLXmty59pci0OEw==", + "version": "3.8.7", + "resolved": "https://registry.npmjs.org/@oclif/parser/-/parser-3.8.7.tgz", + "integrity": "sha512-b11xBmIUK+LuuwVGJpFs4LwQN2xj2cBWj2c4z1FtiXGrJ85h9xV6q+k136Hw0tGg1jQoRXuvuBnqQ7es7vO9/Q==", "dev": true, "requires": { - "@oclif/errors": "^1.2.2", + "@oclif/errors": "^1.3.5", "@oclif/linewrap": "^1.0.0", "chalk": "^4.1.0", - "tslib": "^2.0.0" + "tslib": "^2.3.1" }, "dependencies": { "ansi-styles": { @@ -1703,313 +2066,213 @@ } } }, - "@oclif/plugin-help": { - "version": "2.2.3", - "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-2.2.3.tgz", - "integrity": "sha512-bGHUdo5e7DjPJ0vTeRBMIrfqTRDBfyR5w0MP41u0n3r7YG5p14lvMmiCXxi6WDaP2Hw5nqx3PnkAIntCKZZN7g==", + "@oclif/plugin-autocomplete": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/@oclif/plugin-autocomplete/-/plugin-autocomplete-0.3.0.tgz", + "integrity": "sha512-gCuIUCswvoU1BxDDvHSUGxW8rFagiacle8jHqE49+WnuniXD/N8NmJvnzmlNyc8qLE192CnKK+qYyAF+vaFQBg==", "dev": true, "requires": { "@oclif/command": "^1.5.13", - "chalk": "^2.4.1", - "indent-string": "^4.0.0", - "lodash.template": "^4.4.0", - "string-width": "^3.0.0", - "strip-ansi": "^5.0.0", - "widest-line": "^2.0.1", - "wrap-ansi": "^4.0.0" + "@oclif/config": "^1.13.0", + "chalk": "^4.1.0", + "cli-ux": "^5.2.1", + "debug": "^4.0.0", + "fs-extra": "^9.0.1", + "moment": "^2.22.1" }, "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "emoji-regex": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", - "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", - "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true - }, - "string-width": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", - "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", "dev": true, "requires": { - "emoji-regex": "^7.0.1", - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^5.1.0" + "type-fest": "^0.21.3" } }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", "dev": true, "requires": { - "ansi-regex": "^4.1.0" + "color-convert": "^2.0.1" } }, - "widest-line": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-2.0.1.tgz", - "integrity": "sha512-Ba5m9/Fa4Xt9eb2ELXt77JxVDV8w7qQrH0zS/TWSJdLyAwQjWoOzpzj5lwVftDz6n/EOu3tNACS84v509qwnJA==", - "dev": true, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, "requires": { - "string-width": "^2.1.1" + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "cli-ux": { + "version": "5.6.7", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.7.tgz", + "integrity": "sha512-dsKAurMNyFDnO6X1TiiRNiVbL90XReLKcvIq4H777NMqXGBxBws23ag8ubCJE97vVZEgWG2eSUhsyLf63Jv8+g==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" }, "dependencies": { - "ansi-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", - "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true - }, - "string-width": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", - "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", "dev": true, "requires": { - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^4.0.0" + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" } }, - "strip-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", "dev": true, "requires": { - "ansi-regex": "^3.0.0" + "has-flag": "^4.0.0" } } } }, - "wrap-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-4.0.0.tgz", - "integrity": "sha512-uMTsj9rDb0/7kk1PbcbCcwvHUxp60fGDB/NNXpVa0Q+ic/e7y5+BwTxKfQ33VYgDppSwi/FBzpetYzo8s6tfbg==", + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", "dev": true, "requires": { - "ansi-styles": "^3.2.0", - "string-width": "^2.1.1", - "strip-ansi": "^4.0.0" + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", + "dev": true + }, + "fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "requires": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" }, "dependencies": { - "ansi-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", - "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true - }, - "string-width": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", - "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", "dev": true, "requires": { - "is-fullwidth-code-point": "^2.0.0", - "strip-ansi": "^4.0.0" + "graceful-fs": "^4.1.6", + "universalify": "^2.0.0" } }, - "strip-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", - "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", - "dev": true, - "requires": { - "ansi-regex": "^3.0.0" - } + "universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true } } - } - } - }, - "@oclif/screen": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/@oclif/screen/-/screen-1.0.4.tgz", - "integrity": "sha512-60CHpq+eqnTxLZQ4PGHYNwUX572hgpMHGPtTWMjdTMsAvlm69lZV/4ly6O3sAYkomo4NggGcomrDpBe34rxUqw==", - "dev": true - }, - "@oclif/test": { - "version": "1.2.9", - "resolved": "https://registry.npmjs.org/@oclif/test/-/test-1.2.9.tgz", - "integrity": "sha512-op+ak0NTyeBKqjLVH1jfPCRGWK5befIoQpCL/xwekjucUEmMfCbUpV1Sa60f9rU8X58HEqrclwWbAH+DtQR6FQ==", - "dev": true, - "requires": { - "fancy-test": "^1.4.10" - } - }, - "@prettier/plugin-xml": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@prettier/plugin-xml/-/plugin-xml-1.2.0.tgz", - "integrity": "sha512-bFvVAZKs59XNmntYjyefn3K4TBykS6E+d6ZW8IcylAs88ZO+TzLhp0dPpi0VKfPzq1Nb+kpDnPRTiwb4zY6NgA==", - "dev": true, - "requires": { - "@xml-tools/parser": "^1.0.11", - "prettier": ">=2.3" - } - }, - "@salesforce/bunyan": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/@salesforce/bunyan/-/bunyan-2.0.0.tgz", - "integrity": "sha512-5hq+HWQSeymuygl3i9ehlQo3XWrlBE+A+QzmpDaoK37op4u9M+SBUbXfOW0IABOQCg+JmfQPocSMV74hRoqU9w==", - "dev": true, - "requires": { - "dayjs": "^1.8.16", - "dayjs-plugin-utc": "^0.1.2", - "dtrace-provider": "~0.6", - "mv": "~2", - "safe-json-stringify": "~1" - } - }, - "@salesforce/command": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-3.1.3.tgz", - "integrity": "sha512-Yg9lhl3ghwPN7WwqFmgfWIn6i7vz43WTpEsYsChz80bKORlVbDvhwPZQUj0XTv3DKDnPZVU8FFIDsrLSOa9G1A==", - "dev": true, - "requires": { - "@oclif/command": "^1.5.17", - "@oclif/errors": "^1.2.2", - "@oclif/parser": "^3.8.3", - "@oclif/plugin-help": "^2.2.0", - "@oclif/test": "^1.2.4", - "@salesforce/core": "^2.23.4", - "@salesforce/kit": "^1.2.2", - "@salesforce/ts-types": "^1.2.0", - "chalk": "^2.4.2", - "cli-ux": "^4.9.3" - } - }, - "@salesforce/core": { - "version": "2.33.1", - "resolved": "https://registry.npmjs.org/@salesforce/core/-/core-2.33.1.tgz", - "integrity": "sha512-jKVFYEvlV+loBoau5heBOVXmzsPO+RbYh6SPybJK6xF7khQmzu7+WAQbikY2eY8VaXcded2kka8L/FKuD/LKBg==", - "dev": true, - "requires": { - "@salesforce/bunyan": "^2.0.0", - "@salesforce/kit": "^1.5.0", - "@salesforce/schemas": "^1.0.1", - "@salesforce/ts-types": "^1.5.13", - "@types/graceful-fs": "^4.1.5", - "@types/jsforce": "^1.9.35", - "@types/mkdirp": "^1.0.1", - "debug": "^3.1.0", - "faye": "^1.4.0", - "graceful-fs": "^4.2.4", - "jsen": "0.6.6", - "jsforce": "^1.11.0", - "jsonwebtoken": "8.5.0", - "mkdirp": "1.0.4", - "semver": "^7.3.5", - "ts-retry-promise": "^0.6.0" - }, - "dependencies": { - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", "dev": true, "requires": { - "ms": "^2.1.1" + "lru-cache": "^6.0.0" } }, - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", "dev": true, "requires": { - "lru-cache": "^6.0.0" + "has-flag": "^4.0.0" } - } - } - }, - "@salesforce/eslint-config-lwc": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/@salesforce/eslint-config-lwc/-/eslint-config-lwc-3.2.1.tgz", - "integrity": "sha512-tX+HKTkgEEYSI/cggkeUkRT+m0Hrv3FmxLclRMjUZWDZQYggyLzpWkGX9Nkj69xRssHJ0N/7pE4UNlbqCNpbrQ==", - "dev": true, - "requires": { - "@babel/core": "~7.16.0", - "@babel/eslint-parser": "~7.16.0", - "eslint-restricted-globals": "~0.2.0", - "semver": "^7.3.5" - }, - "dependencies": { - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", "dev": true, "requires": { - "lru-cache": "^6.0.0" + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" } } } }, - "@salesforce/eslint-plugin-aura": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/@salesforce/eslint-plugin-aura/-/eslint-plugin-aura-2.0.0.tgz", - "integrity": "sha512-SLLnPk2fl1HIccPJgB5d0FCmWX9tgBKiK6sINUVixDgFcXLzogrnOmv5oCbXKbcJuOIoeG/JJDNjcb/WbsyvgQ==", - "dev": true, - "requires": { - "eslint-plugin-compat": "^3.9.0" - } - }, - "@salesforce/eslint-plugin-lightning": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@salesforce/eslint-plugin-lightning/-/eslint-plugin-lightning-1.0.0.tgz", - "integrity": "sha512-zk0PKXAcHKHepAG2EOSWlkOTxQM0Aw1CT6+MUxJcM42fCDwH/yPPpGkG3CWtRfmVViODGOwU9ywU2wlkAYcvUQ==", - "dev": true - }, - "@salesforce/kit": { - "version": "1.5.25", - "resolved": "https://registry.npmjs.org/@salesforce/kit/-/kit-1.5.25.tgz", - "integrity": "sha512-Tbb7AZwJ00oGW8uv4DWsb3tjYi/rI8XWICWhLDpi44lTjrd+b22hKZMJLZ6XWEmezwtbzLT7vZIcj3IMtDgkIg==", - "dev": true, - "requires": { - "@salesforce/ts-types": "^1.5.20", - "shx": "^0.3.3", - "tslib": "^2.2.0" - } - }, - "@salesforce/schemas": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@salesforce/schemas/-/schemas-1.1.0.tgz", - "integrity": "sha512-6D7DvE6nFxpLyyTnrOIbbAeCJw2r/EpinFAcMh6gU0gA/CGfSbwV/8uR3uHLYL2zCyCZLH8jJ4dZ3BzCMqc+Eg==", - "dev": true - }, - "@salesforce/sfdx-lwc-jest": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/@salesforce/sfdx-lwc-jest/-/sfdx-lwc-jest-1.1.0.tgz", - "integrity": "sha512-bYhvs9Th4qMjk4Vs66GDwloVh2LAcV8cg/j0WWbIug8MiI+KOSZL0IWZA7fqr8vXkSdArt5WjoTv4ckzfklzQA==", + "@oclif/plugin-commands": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@oclif/plugin-commands/-/plugin-commands-1.3.0.tgz", + "integrity": "sha512-Qx9gJ7/aPBgo+Q/DHmGcWyxn2/0bjqmCwt/nO0lWuTZQIH3ZTqclTm68TMZLS4QnQyDGeeYK0GqZ5qJlrXD+SQ==", "dev": true, "requires": { - "@lwc/compiler": "2.2.11", - "@lwc/engine-dom": "2.2.11", - "@lwc/jest-preset": "11.2.0", - "@lwc/jest-resolver": "11.2.0", - "@lwc/jest-serializer": "11.2.0", - "@lwc/jest-transformer": "11.2.0", - "@lwc/module-resolver": "2.2.11", - "@lwc/synthetic-shadow": "2.2.11", - "@lwc/wire-service": "2.2.11", - "@salesforce/wire-service-jest-util": "4.0.0", - "chalk": "~4.1.1", - "fast-glob": "^3.2.7", - "jest": "27.0.6", - "yargs": "~17.0.1" + "@oclif/command": "^1.5.4", + "@oclif/config": "^1.8.7", + "cli-ux": "^5.4.5", + "lodash": "^4.17.11" }, "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, "ansi-styles": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", @@ -2027,6 +2290,51 @@ "requires": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "cli-ux": { + "version": "5.6.7", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.7.tgz", + "integrity": "sha512-dsKAurMNyFDnO6X1TiiRNiVbL90XReLKcvIq4H777NMqXGBxBws23ag8ubCJE97vVZEgWG2eSUhsyLf63Jv8+g==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" } }, "color-convert": { @@ -2044,1011 +2352,904 @@ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", "dev": true, "requires": { "has-flag": "^4.0.0" } }, - "yargs": { - "version": "17.0.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.0.1.tgz", - "integrity": "sha512-xBBulfCc8Y6gLFcrPvtqKz9hz8SO0l1Ni8GgDekvBX2ro0HRQImDGnikfc33cgzcYUSncapnNcZDjVFIH3f6KQ==", + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", "dev": true, "requires": { - "cliui": "^7.0.2", - "escalade": "^3.1.1", - "get-caller-file": "^2.0.5", - "require-directory": "^2.1.1", - "string-width": "^4.2.0", - "y18n": "^5.0.5", - "yargs-parser": "^20.2.2" + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } } } } }, - "@salesforce/sfdx-scanner": { - "version": "2.12.0", - "resolved": "https://registry.npmjs.org/@salesforce/sfdx-scanner/-/sfdx-scanner-2.12.0.tgz", - "integrity": "sha512-0v7hA9c1y7DUUbbAAGA2ngr9cqQVUbiUgoaXCKoQYMFfEV2aSLft5MjAJ377lUNV8pjwPzeId2ndvIscNoZX/g==", + "@oclif/plugin-help": { + "version": "2.2.3", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-2.2.3.tgz", + "integrity": "sha512-bGHUdo5e7DjPJ0vTeRBMIrfqTRDBfyR5w0MP41u0n3r7YG5p14lvMmiCXxi6WDaP2Hw5nqx3PnkAIntCKZZN7g==", "dev": true, "requires": { - "@lwc/eslint-plugin-lwc": "^0.10.0", - "@oclif/command": "^1", - "@oclif/config": "^1", - "@oclif/errors": "^1", - "@salesforce/command": "^3", - "@salesforce/core": "^2.1.6", - "@salesforce/eslint-config-lwc": "^0.7.0", - "@typescript-eslint/eslint-plugin": "^2.21.0", - "@typescript-eslint/parser": "^2.21.0", - "babel-eslint": "^10.1.0", - "cross-spawn": "^7.0.3", - "eslint": "^6.8.0", - "find-java-home": "1.1.0", - "globby": "^11.0.0", - "html-escaper": "^3.0.0", - "is-zip": "^1.0.0", - "isbinaryfile": "^4.0.8", - "mustache": "^4.0.1", - "node-stream-zip": "1.13.2", - "normalize-path": "^3.0.0", - "picomatch": "^2.2.2", - "reflect-metadata": "^0.1.13", - "retire": "^2.2.5", - "semver": "^7.3.4", - "ts-node": "^8", - "tslib": "^2", - "tsyringe": "^4.1.0", - "typescript": "^3.8.2", - "untildify": "^4.0.0", - "word-wrap": "^1.2.3", - "xml-js": "^1.6.11" + "@oclif/command": "^1.5.13", + "chalk": "^2.4.1", + "indent-string": "^4.0.0", + "lodash.template": "^4.4.0", + "string-width": "^3.0.0", + "strip-ansi": "^5.0.0", + "widest-line": "^2.0.1", + "wrap-ansi": "^4.0.0" }, "dependencies": { - "@lwc/eslint-plugin-lwc": { - "version": "0.10.0", - "resolved": "https://registry.npmjs.org/@lwc/eslint-plugin-lwc/-/eslint-plugin-lwc-0.10.0.tgz", - "integrity": "sha512-UsosUow0xWrzzcSK1fWDWHKBT81/pJz1/icfv7w8T+BCE9G4Kb4TwJexHLOIZvY7e/dv4Fjme4lTOWehuNE5Sg==", - "dev": true, - "requires": { - "minimatch": "^3.0.4" - } + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true }, - "@salesforce/eslint-config-lwc": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/@salesforce/eslint-config-lwc/-/eslint-config-lwc-0.7.0.tgz", - "integrity": "sha512-x87H9+amXcPDmOnOZoLdajp5yNZqv8ieyzfE4R8QDOJeZfj1X5I2AOSZYG0pUV51eUV/kOdGjAj28zU296t2yQ==", + "emoji-regex": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", + "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "string-width": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", + "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", "dev": true, "requires": { - "@lwc/eslint-plugin-lwc": "~0.10.0", - "babel-eslint": "~10.1.0", - "eslint-plugin-import": "~2.20.2", - "eslint-plugin-jest": "~23.8.2", - "eslint-restricted-globals": "~0.2.0" + "emoji-regex": "^7.0.1", + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^5.1.0" } }, - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", "dev": true, "requires": { - "ms": "2.0.0" + "ansi-regex": "^4.1.0" } }, - "eslint": { - "version": "6.8.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-6.8.0.tgz", - "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", + "widest-line": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-2.0.1.tgz", + "integrity": "sha512-Ba5m9/Fa4Xt9eb2ELXt77JxVDV8w7qQrH0zS/TWSJdLyAwQjWoOzpzj5lwVftDz6n/EOu3tNACS84v509qwnJA==", "dev": true, "requires": { - "@babel/code-frame": "^7.0.0", - "ajv": "^6.10.0", - "chalk": "^2.1.0", - "cross-spawn": "^6.0.5", - "debug": "^4.0.1", - "doctrine": "^3.0.0", - "eslint-scope": "^5.0.0", - "eslint-utils": "^1.4.3", - "eslint-visitor-keys": "^1.1.0", - "espree": "^6.1.2", - "esquery": "^1.0.1", - "esutils": "^2.0.2", - "file-entry-cache": "^5.0.1", - "functional-red-black-tree": "^1.0.1", - "glob-parent": "^5.0.0", - "globals": "^12.1.0", - "ignore": "^4.0.6", - "import-fresh": "^3.0.0", - "imurmurhash": "^0.1.4", - "inquirer": "^7.0.0", - "is-glob": "^4.0.0", - "js-yaml": "^3.13.1", - "json-stable-stringify-without-jsonify": "^1.0.1", - "levn": "^0.3.0", - "lodash": "^4.17.14", - "minimatch": "^3.0.4", - "mkdirp": "^0.5.1", - "natural-compare": "^1.4.0", - "optionator": "^0.8.3", - "progress": "^2.0.0", - "regexpp": "^2.0.1", - "semver": "^6.1.2", - "strip-ansi": "^5.2.0", - "strip-json-comments": "^3.0.1", - "table": "^5.2.3", - "text-table": "^0.2.0", - "v8-compile-cache": "^2.0.3" + "string-width": "^2.1.1" }, "dependencies": { - "cross-spawn": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", - "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", - "dev": true, - "requires": { - "nice-try": "^1.0.4", - "path-key": "^2.0.1", - "semver": "^5.5.0", - "shebang-command": "^1.2.0", - "which": "^1.2.9" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } - } + "ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true }, - "debug": { - "version": "4.3.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", - "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", "dev": true, "requires": { - "ms": "2.1.2" + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" } }, - "doctrine": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", - "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { - "esutils": "^2.0.2" + "ansi-regex": "^3.0.0" } - }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, - "semver": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true } } }, - "eslint-plugin-import": { - "version": "2.20.2", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.20.2.tgz", - "integrity": "sha512-FObidqpXrR8OnCh4iNsxy+WACztJLXAHBO5hK79T1Hc77PgQZkyDGA5Ag9xAvRpglvLNxhH/zSmZ70/pZ31dHg==", + "wrap-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-4.0.0.tgz", + "integrity": "sha512-uMTsj9rDb0/7kk1PbcbCcwvHUxp60fGDB/NNXpVa0Q+ic/e7y5+BwTxKfQ33VYgDppSwi/FBzpetYzo8s6tfbg==", "dev": true, "requires": { - "array-includes": "^3.0.3", - "array.prototype.flat": "^1.2.1", - "contains-path": "^0.1.0", - "debug": "^2.6.9", - "doctrine": "1.5.0", - "eslint-import-resolver-node": "^0.3.2", - "eslint-module-utils": "^2.4.1", - "has": "^1.0.3", - "minimatch": "^3.0.4", - "object.values": "^1.1.0", - "read-pkg-up": "^2.0.0", - "resolve": "^1.12.0" + "ansi-styles": "^3.2.0", + "string-width": "^2.1.1", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + } + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + } + } + }, + "@oclif/plugin-not-found": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/@oclif/plugin-not-found/-/plugin-not-found-1.2.6.tgz", + "integrity": "sha512-cfkDub79I9EpselfU/W8FTXhslrkOgfqjaa25tyGo99dAX5UVr6BWL2wbUobsU+rUcm4HN3byzdHDcqfu6hoAw==", + "dev": true, + "requires": { + "@oclif/color": "^0.1.2", + "@oclif/command": "1.8.11", + "cli-ux": "5.6.6", + "fast-levenshtein": "^3.0.0", + "lodash": "^4.17.21" + }, + "dependencies": { + "@oclif/command": { + "version": "1.8.11", + "resolved": "https://registry.npmjs.org/@oclif/command/-/command-1.8.11.tgz", + "integrity": "sha512-2fGLMvi6J5+oNxTaZfdWPMWY8oW15rYj0V8yLzmZBAEjfzjLqLIzJE9IlNccN1zwRqRHc1bcISSRDdxJ56IS/Q==", + "dev": true, + "requires": { + "@oclif/config": "^1.18.2", + "@oclif/errors": "^1.3.5", + "@oclif/parser": "^3.8.6", + "@oclif/plugin-help": "3.2.14", + "debug": "^4.1.1", + "semver": "^7.3.2" } }, - "eslint-utils": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz", - "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", + "@oclif/plugin-help": { + "version": "3.2.14", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-3.2.14.tgz", + "integrity": "sha512-NP5qmE2YfcW3MmXjcrxiqKe9Hf3G0uK/qNc0zAMYKU4crFyIsWj7dBfQVFZSb28YXGioOOpjMzG1I7VMxKF38Q==", "dev": true, "requires": { - "eslint-visitor-keys": "^1.1.0" + "@oclif/command": "^1.8.9", + "@oclif/config": "^1.18.2", + "@oclif/errors": "^1.3.5", + "chalk": "^4.1.2", + "indent-string": "^4.0.0", + "lodash": "^4.17.21", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "widest-line": "^3.1.0", + "wrap-ansi": "^6.2.0" } }, - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", - "dev": true + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } }, - "globals": { - "version": "12.4.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", - "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", "dev": true, "requires": { - "type-fest": "^0.8.1" + "color-convert": "^2.0.1" } }, - "html-escaper": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-3.0.3.tgz", - "integrity": "sha512-RuMffC89BOWQoY0WKGpIhn5gX3iI54O6nRA0yC124NYVtzjmFWBIiFd8M0x+ZdX0P9R4lADg1mgP8C7PxGOWuQ==", - "dev": true + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } }, - "ignore": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", - "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", - "dev": true + "cli-ux": { + "version": "5.6.6", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.6.tgz", + "integrity": "sha512-4wUB34zoFklcZV0z5YiOM5IqVMMt9c3TK3QYRK3dqyk3XoRC0ybiWDWHfsMDjkKrzsVTw95rXn9NrzSHbae4pg==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.9", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", "dev": true, "requires": { - "minimist": "^1.2.5" + "color-name": "~1.1.4" } }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, - "path-key": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", - "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", "dev": true }, - "regexpp": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", - "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "fast-levenshtein": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-3.0.0.tgz", + "integrity": "sha512-hKKNajm46uNmTlhHSyZkmToAc56uZJwYq7yrciZjqOxnlfQwERDQJmHPUp7m1m9wx8vgOe8IaCKZ5Kv2k1DdCQ==", + "dev": true, + "requires": { + "fastest-levenshtein": "^1.0.7" + } + }, + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true }, "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", "dev": true, "requires": { "lru-cache": "^6.0.0" } }, - "shebang-command": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", - "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", "dev": true, "requires": { - "shebang-regex": "^1.0.0" + "has-flag": "^4.0.0" } }, - "shebang-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", - "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", - "dev": true - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", "dev": true, "requires": { - "ansi-regex": "^4.1.0" + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" } }, - "type-fest": { - "version": "0.8.1", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", - "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", - "dev": true - }, - "which": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", - "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", "dev": true, "requires": { - "isexe": "^2.0.0" + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" } } } }, - "@salesforce/ts-types": { - "version": "1.5.20", - "resolved": "https://registry.npmjs.org/@salesforce/ts-types/-/ts-types-1.5.20.tgz", - "integrity": "sha512-Ov6um4CWd63EvkRavkHG0J/P9XYL55sdkDWPMr7+AIgqh5flHxDRz09/C4e9M94aX30rzJxW4TVX6EBf4Cu2BQ==", - "dev": true, - "requires": { - "tslib": "^2.2.0" - } - }, - "@salesforce/wire-service-jest-util": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/@salesforce/wire-service-jest-util/-/wire-service-jest-util-4.0.0.tgz", - "integrity": "sha512-YJO/bMq5l6IYIZG6bAqYzzbmZMPCzB2GE2TKGLA1U7B9HpmNHZS7DdDcy154P03dfLSgF+tgVYeklh2HRYGk9g==", - "dev": true - }, - "@sideway/address": { - "version": "4.1.3", - "resolved": "https://registry.npmjs.org/@sideway/address/-/address-4.1.3.tgz", - "integrity": "sha512-8ncEUtmnTsMmL7z1YPB47kPUq7LpKWJNFPsRzHiIajGC5uXlWGn+AmkYPcHNl8S4tcEGx+cnORnNYaw2wvL+LQ==", - "dev": true, - "requires": { - "@hapi/hoek": "^9.0.0" - } - }, - "@sideway/formula": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@sideway/formula/-/formula-3.0.0.tgz", - "integrity": "sha512-vHe7wZ4NOXVfkoRb8T5otiENVlT7a3IAiw7H5M2+GO+9CDgcVUUsX1zalAztCmwyOr2RUTGJdgB+ZvSVqmdHmg==", - "dev": true - }, - "@sideway/pinpoint": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/@sideway/pinpoint/-/pinpoint-2.0.0.tgz", - "integrity": "sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ==", - "dev": true - }, - "@sinonjs/commons": { - "version": "1.8.3", - "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.3.tgz", - "integrity": "sha512-xkNcLAn/wZaX14RPlwizcKicDk9G3F8m2nU3L7Ukm5zBgTwiT0wsoFAHx9Jq56fJA1z/7uKGtCRu16sOUCLIHQ==", - "dev": true, - "requires": { - "type-detect": "4.0.8" - } - }, - "@sinonjs/fake-timers": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-8.1.0.tgz", - "integrity": "sha512-OAPJUAtgeINhh/TAlUID4QTs53Njm7xzddaVlEs/SXwgtiD1tW22zAB/W1wdqfrpmikgaWQ9Fw6Ws+hsiRm5Vg==", - "dev": true, - "requires": { - "@sinonjs/commons": "^1.7.0" - } - }, - "@tootallnate/once": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", - "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", - "dev": true - }, - "@types/babel__core": { - "version": "7.1.18", - "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.18.tgz", - "integrity": "sha512-S7unDjm/C7z2A2R9NzfKCK1I+BAALDtxEmsJBwlB3EzNfb929ykjL++1CK9LO++EIp2fQrC8O+BwjKvz6UeDyQ==", - "dev": true, - "requires": { - "@babel/parser": "^7.1.0", - "@babel/types": "^7.0.0", - "@types/babel__generator": "*", - "@types/babel__template": "*", - "@types/babel__traverse": "*" - } - }, - "@types/babel__generator": { - "version": "7.6.4", - "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.4.tgz", - "integrity": "sha512-tFkciB9j2K755yrTALxD44McOrk+gfpIpvC3sxHjRawj6PfnQxrse4Clq5y/Rq+G3mrBurMax/lG8Qn2t9mSsg==", - "dev": true, - "requires": { - "@babel/types": "^7.0.0" - } - }, - "@types/babel__template": { - "version": "7.4.1", - "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.1.tgz", - "integrity": "sha512-azBFKemX6kMg5Io+/rdGT0dkGreboUVR0Cdm3fz9QJWpaQGJRQXl7C+6hOTCZcMll7KFyEQpgbYI2lHdsS4U7g==", + "@oclif/plugin-plugins": { + "version": "1.10.11", + "resolved": "https://registry.npmjs.org/@oclif/plugin-plugins/-/plugin-plugins-1.10.11.tgz", + "integrity": "sha512-C9eHF10UkxwoAqRYrPW51YDuDOpDXASX4BEA++kTVcqhMQTKBQalmEJKw+gVnLl1YNmapse1ZSAcU1TrXjqykg==", "dev": true, "requires": { - "@babel/parser": "^7.1.0", - "@babel/types": "^7.0.0" - } - }, - "@types/babel__traverse": { - "version": "7.14.2", - "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.14.2.tgz", - "integrity": "sha512-K2waXdXBi2302XUdcHcR1jCeU0LL4TD9HRs/gk0N2Xvrht+G/BfJa4QObBQZfhMdxiCpV3COl5Nfq4uKTeTnJA==", - "dev": true, - "requires": { - "@babel/types": "^7.3.0" - } - }, - "@types/chai": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.3.0.tgz", - "integrity": "sha512-/ceqdqeRraGolFTcfoXNiqjyQhZzbINDngeoAq9GoHa8PPK1yNzTaxWjA6BFWp5Ua9JpXEMSS4s5i9tS0hOJtw==", - "dev": true - }, - "@types/eslint-visitor-keys": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@types/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", - "integrity": "sha512-OCutwjDZ4aFS6PB1UZ988C4YgwlBHJd6wCeQqaLdmadZ/7e+w79+hbMUFC1QXDNCmdyoRfAFdm0RypzwR+Qpag==", - "dev": true - }, - "@types/graceful-fs": { - "version": "4.1.5", - "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.5.tgz", - "integrity": "sha512-anKkLmZZ+xm4p8JWBf4hElkM4XR+EZeA2M9BAkkTldmcyDY4mbdIJnRghDJH3Ov5ooY7/UAoENtmdMSkaAd7Cw==", - "dev": true, - "requires": { - "@types/node": "*" - } - }, - "@types/istanbul-lib-coverage": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz", - "integrity": "sha512-z/QT1XN4K4KYuslS23k62yDIDLwLFkzxOuMplDtObz0+y7VqJCaO2o+SPwHCvLFZh7xazvvoor2tA/hPz9ee7g==", - "dev": true - }, - "@types/istanbul-lib-report": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", - "integrity": "sha512-plGgXAPfVKFoYfa9NpYDAkseG+g6Jr294RqeqcqDixSbU34MZVJRi/P+7Y8GDpzkEwLaGZZOpKIEmeVZNtKsrg==", - "dev": true, - "requires": { - "@types/istanbul-lib-coverage": "*" - } - }, - "@types/istanbul-reports": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.1.tgz", - "integrity": "sha512-c3mAZEuK0lvBp8tmuL74XRKn1+y2dcwOUpH7x4WrF6gk1GIgiluDRgMYQtw2OFcBvAJWlt6ASU3tSqxp0Uu0Aw==", - "dev": true, - "requires": { - "@types/istanbul-lib-report": "*" - } - }, - "@types/jsforce": { - "version": "1.9.38", - "resolved": "https://registry.npmjs.org/@types/jsforce/-/jsforce-1.9.38.tgz", - "integrity": "sha512-+Iwf5jlDiK8z+zI2LAi4mzln8++5lETv2YofFEATu+dNkrP8LACB76lz2tPsXDx/a+5uW8HQhbwL/SyqIic0cg==", - "dev": true, - "requires": { - "@types/node": "*" - } - }, - "@types/json-schema": { - "version": "7.0.9", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.9.tgz", - "integrity": "sha512-qcUXuemtEu+E5wZSJHNxUXeCZhAfXKQ41D+duX+VYPde7xyEVZci+/oXKJL13tnRs9lR2pr4fod59GT6/X1/yQ==", - "dev": true - }, - "@types/json5": { - "version": "0.0.29", - "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", - "integrity": "sha1-7ihweulOEdK4J7y+UnC86n8+ce4=", - "dev": true - }, - "@types/lodash": { - "version": "4.14.178", - "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.178.tgz", - "integrity": "sha512-0d5Wd09ItQWH1qFbEyQ7oTQ3GZrMfth5JkbN3EvTKLXcHLRDSXeLnlvlOn0wvxVIwK5o2M8JzP/OWz7T3NRsbw==", - "dev": true - }, - "@types/mkdirp": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/@types/mkdirp/-/mkdirp-1.0.2.tgz", - "integrity": "sha512-o0K1tSO0Dx5X6xlU5F1D6625FawhC3dU3iqr25lluNv/+/QIVH8RLNEiVokgIZo+mz+87w/3Mkg/VvQS+J51fQ==", - "dev": true, - "requires": { - "@types/node": "*" - } - }, - "@types/node": { - "version": "6.14.13", - "resolved": "https://registry.npmjs.org/@types/node/-/node-6.14.13.tgz", - "integrity": "sha512-J1F0XJ/9zxlZel5ZlbeSuHW2OpabrUAqpFuC2sm2I3by8sERQ8+KCjNKUcq8QHuzpGMWiJpo9ZxeHrqrP2KzQw==", - "dev": true - }, - "@types/prettier": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.4.2.tgz", - "integrity": "sha512-ekoj4qOQYp7CvjX8ZDBgN86w3MqQhLE1hczEJbEIjgFEumDy+na/4AJAbLXfgEWFNB2pKadM5rPFtuSGMWK7xA==", - "dev": true - }, - "@types/sinon": { - "version": "10.0.6", - "resolved": "https://registry.npmjs.org/@types/sinon/-/sinon-10.0.6.tgz", - "integrity": "sha512-6EF+wzMWvBNeGrfP3Nx60hhx+FfwSg1JJBLAAP/IdIUq0EYkqCYf70VT3PhuhPX9eLD+Dp+lNdpb/ZeHG8Yezg==", - "dev": true, - "requires": { - "@sinonjs/fake-timers": "^7.1.0" + "@oclif/color": "^0.1.2", + "@oclif/command": "^1.8.15", + "@oclif/errors": "^1.3.5", + "chalk": "^4.1.2", + "cli-ux": "^5.6.7", + "debug": "^4.3.3", + "fs-extra": "^9.0", + "http-call": "^5.3.0", + "load-json-file": "^5.3.0", + "npm-run-path": "^4.0.1", + "semver": "^7.3.2", + "tslib": "^2.0.0", + "yarn": "^1.21.1" }, "dependencies": { - "@sinonjs/fake-timers": { - "version": "7.1.2", - "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-7.1.2.tgz", - "integrity": "sha512-iQADsW4LBMISqZ6Ci1dupJL9pprqwcVFTcOsEmQOEhW+KLCVn/Y4Jrvg2k19fIHCp+iFprriYPTdRcQR8NbUPg==", + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", "dev": true, "requires": { - "@sinonjs/commons": "^1.7.0" + "type-fest": "^0.21.3" } - } - } - }, - "@types/stack-utils": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.1.tgz", - "integrity": "sha512-Hl219/BT5fLAaz6NDkSuhzasy49dwQS/DSdu4MdggFB8zcXv7vflBI3xp7FEmkmdDkBUI2bPUNeMttp2knYdxw==", - "dev": true - }, - "@types/yargs": { - "version": "16.0.4", - "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-16.0.4.tgz", - "integrity": "sha512-T8Yc9wt/5LbJyCaLiHPReJa0kApcIgJ7Bn735GjItUfh08Z1pJvu8QZqb9s+mMvKV6WUQRV7K2R46YbjMXTTJw==", - "dev": true, - "requires": { - "@types/yargs-parser": "*" - } - }, - "@types/yargs-parser": { - "version": "20.2.1", - "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-20.2.1.tgz", - "integrity": "sha512-7tFImggNeNBVMsn0vLrpn1H1uPrUBdnARPTpZoitY37ZrdJREzf7I16tMrlK3hen349gr1NYh8CmZQa7CTG6Aw==", - "dev": true - }, - "@typescript-eslint/eslint-plugin": { - "version": "2.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-2.34.0.tgz", - "integrity": "sha512-4zY3Z88rEE99+CNvTbXSyovv2z9PNOVffTWD2W8QF5s2prBQtwN2zadqERcrHpcR7O/+KMI3fcTAmUUhK/iQcQ==", - "dev": true, - "requires": { - "@typescript-eslint/experimental-utils": "2.34.0", - "functional-red-black-tree": "^1.0.1", - "regexpp": "^3.0.0", - "tsutils": "^3.17.1" - } - }, - "@typescript-eslint/experimental-utils": { - "version": "2.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-2.34.0.tgz", - "integrity": "sha512-eS6FTkq+wuMJ+sgtuNTtcqavWXqsflWcfBnlYhg/nS4aZ1leewkXGbvBhaapn1q6qf4M71bsR1tez5JTRMuqwA==", - "dev": true, - "requires": { - "@types/json-schema": "^7.0.3", - "@typescript-eslint/typescript-estree": "2.34.0", - "eslint-scope": "^5.0.0", - "eslint-utils": "^2.0.0" - } - }, - "@typescript-eslint/parser": { - "version": "2.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-2.34.0.tgz", - "integrity": "sha512-03ilO0ucSD0EPTw2X4PntSIRFtDPWjrVq7C3/Z3VQHRC7+13YB55rcJI3Jt+YgeHbjUdJPcPa7b23rXCBokuyA==", - "dev": true, - "requires": { - "@types/eslint-visitor-keys": "^1.0.0", - "@typescript-eslint/experimental-utils": "2.34.0", - "@typescript-eslint/typescript-estree": "2.34.0", - "eslint-visitor-keys": "^1.1.0" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "cli-ux": { + "version": "5.6.7", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.7.tgz", + "integrity": "sha512-dsKAurMNyFDnO6X1TiiRNiVbL90XReLKcvIq4H777NMqXGBxBws23ag8ubCJE97vVZEgWG2eSUhsyLf63Jv8+g==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true - } - } - }, - "@typescript-eslint/typescript-estree": { - "version": "2.34.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-2.34.0.tgz", - "integrity": "sha512-OMAr+nJWKdlVM9LOqCqh3pQQPwxHAN7Du8DR6dmwCrAmxtiXQnhHJ6tBNtf+cggqfo51SG/FCwnKhXCIM7hnVg==", - "dev": true, - "requires": { - "debug": "^4.1.1", - "eslint-visitor-keys": "^1.1.0", - "glob": "^7.1.6", - "is-glob": "^4.0.1", - "lodash": "^4.17.15", - "semver": "^7.3.2", - "tsutils": "^3.17.1" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + }, + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", "dev": true }, - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", "dev": true, "requires": { - "lru-cache": "^6.0.0" + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "dependencies": { + "jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6", + "universalify": "^2.0.0" + } + }, + "universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true + } } - } - } - }, - "@xml-tools/parser": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/@xml-tools/parser/-/parser-1.0.11.tgz", - "integrity": "sha512-aKqQ077XnR+oQtHJlrAflaZaL7qZsulWc/i/ZEooar5JiWj1eLt0+Wg28cpa+XLney107wXqneC+oG1IZvxkTA==", - "dev": true, - "requires": { - "chevrotain": "7.1.1" - } - }, - "abab": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.5.tgz", - "integrity": "sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q==", - "dev": true - }, - "acorn": { - "version": "8.4.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.4.1.tgz", - "integrity": "sha512-asabaBSkEKosYKMITunzX177CXxQ4Q8BSSzMTKD+FefUhipQC70gfW5SiUDhYQ3vk8G+81HqQk7Fv9OXwwn9KA==", - "dev": true - }, - "acorn-globals": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-6.0.0.tgz", - "integrity": "sha512-ZQl7LOWaF5ePqqcX4hLuv/bLXYQNfNWw2c0/yX/TsPRKamzHcTGQnlCjHT3TsmkOUVEPS3crCxiPfdzE/Trlhg==", - "dev": true, - "requires": { - "acorn": "^7.1.1", - "acorn-walk": "^7.1.1" - }, - "dependencies": { - "acorn": { - "version": "7.4.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", - "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "load-json-file": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-5.3.0.tgz", + "integrity": "sha512-cJGP40Jc/VXUsp8/OrnyKyTZ1y6v/dphm3bioS+RrKXjK2BB6wHUd6JptZEFDGgGahMT+InnZO5i1Ei9mpC8Bw==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.15", + "parse-json": "^4.0.0", + "pify": "^4.0.1", + "strip-bom": "^3.0.0", + "type-fest": "^0.3.0" + }, + "dependencies": { + "type-fest": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.3.1.tgz", + "integrity": "sha512-cUGJnCdr4STbePCgqNFbpVNCepa+kAVohJs1sLhxzdH+gnEoOd8VhbYa7pD3zZYGiURWM2xzEII3fQcRizDkYQ==", + "dev": true + } + } + }, + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + } } } }, - "acorn-jsx": { - "version": "5.3.2", - "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", - "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", - "dev": true - }, - "acorn-walk": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz", - "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", - "dev": true - }, - "agent-base": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", - "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", - "dev": true, - "requires": { - "debug": "4" - } - }, - "aggregate-error": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", - "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", + "@oclif/plugin-update": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@oclif/plugin-update/-/plugin-update-1.5.0.tgz", + "integrity": "sha512-GsWK1CMeBBO8YknThoOZulj3xE+ZgZAXW1ouNJALXcs3mbROzszLDGjXV3RM6ffbJpnWLiMIqSFNOE8d+vGcgQ==", "dev": true, "requires": { - "clean-stack": "^2.0.0", - "indent-string": "^4.0.0" + "@oclif/color": "^0.1.0", + "@oclif/command": "^1.7.0", + "@oclif/config": "^1.16.0", + "@oclif/errors": "^1.3.4", + "@types/semver": "^7.3.4", + "cli-ux": "^5.5.1", + "cross-spawn": "^7.0.3", + "debug": "^4.3.1", + "filesize": "^6.1.0", + "fs-extra": "^9.0.1", + "http-call": "^5.3.0", + "lodash": "^4.17.21", + "log-chopper": "^1.0.2", + "semver": "^7.3.5", + "tar-fs": "^2.1.1" }, "dependencies": { - "clean-stack": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", - "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "cli-ux": { + "version": "5.6.7", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.7.tgz", + "integrity": "sha512-dsKAurMNyFDnO6X1TiiRNiVbL90XReLKcvIq4H777NMqXGBxBws23ag8ubCJE97vVZEgWG2eSUhsyLf63Jv8+g==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + } + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", + "dev": true + }, + "fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "requires": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "dependencies": { + "jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6", + "universalify": "^2.0.0" + } + }, + "universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true + } + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", "dev": true + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } } } }, - "ajv": { - "version": "6.12.6", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", - "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "@oclif/plugin-warn-if-update-available": { + "version": "1.7.3", + "resolved": "https://registry.npmjs.org/@oclif/plugin-warn-if-update-available/-/plugin-warn-if-update-available-1.7.3.tgz", + "integrity": "sha512-q8q0NIneVCwIAJzglUMsl3EbXR/H5aPDk6g+qs7uF0tToxe07SWSONoNaKPzViwRWvYChMPjL77/rXyW1HVn4A==", "dev": true, "requires": { - "fast-deep-equal": "^3.1.1", - "fast-json-stable-stringify": "^2.0.0", - "json-schema-traverse": "^0.4.1", - "uri-js": "^4.2.2" - } - }, - "ansi-escape-sequences": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-escape-sequences/-/ansi-escape-sequences-4.1.0.tgz", - "integrity": "sha512-dzW9kHxH011uBsidTXd14JXgzye/YLb2LzeKZ4bsgl/Knwx8AtbSFkkGxagdNOoh0DlqHCmfiEjWKBaqjOanVw==", - "dev": true, - "requires": { - "array-back": "^3.0.1" - }, - "dependencies": { - "array-back": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz", - "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==", - "dev": true - } - } - }, - "ansi-escapes": { - "version": "4.3.2", - "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", - "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", - "dev": true, - "requires": { - "type-fest": "^0.21.3" - } - }, - "ansi-regex": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", - "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", - "dev": true - }, - "ansi-styles": { - "version": "3.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", - "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", - "dev": true, - "requires": { - "color-convert": "^1.9.0" - } - }, - "ansicolors": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/ansicolors/-/ansicolors-0.3.2.tgz", - "integrity": "sha1-ZlWX3oap/+Oqm/vmyuXG6kJrSXk=", - "dev": true - }, - "anymatch": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", - "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", - "dev": true, - "requires": { - "normalize-path": "^3.0.0", - "picomatch": "^2.0.4" - } - }, - "archiver": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/archiver/-/archiver-5.3.0.tgz", - "integrity": "sha512-iUw+oDwK0fgNpvveEsdQ0Ase6IIKztBJU2U0E9MzszMfmVVUyv1QJhS2ITW9ZCqx8dktAxVAjWWkKehuZE8OPg==", - "dev": true, - "requires": { - "archiver-utils": "^2.1.0", - "async": "^3.2.0", - "buffer-crc32": "^0.2.1", - "readable-stream": "^3.6.0", - "readdir-glob": "^1.0.0", - "tar-stream": "^2.2.0", - "zip-stream": "^4.1.0" - }, - "dependencies": { - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - } - } - }, - "archiver-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/archiver-utils/-/archiver-utils-2.1.0.tgz", - "integrity": "sha512-bEL/yUb/fNNiNTuUz979Z0Yg5L+LzLxGJz8x79lYmR54fmTIb6ob/hNQgkQnIUDWIFjZVQwl9Xs356I6BAMHfw==", - "dev": true, - "requires": { - "glob": "^7.1.4", - "graceful-fs": "^4.2.0", - "lazystream": "^1.0.0", - "lodash.defaults": "^4.2.0", - "lodash.difference": "^4.5.0", - "lodash.flatten": "^4.4.0", - "lodash.isplainobject": "^4.0.6", - "lodash.union": "^4.6.0", - "normalize-path": "^3.0.0", - "readable-stream": "^2.0.0" - } - }, - "arg": { - "version": "4.1.3", - "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", - "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", - "dev": true - }, - "argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "dev": true, - "requires": { - "sprintf-js": "~1.0.2" - } - }, - "array-back": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-6.2.0.tgz", - "integrity": "sha512-mixVv03GOOn/ubHE4STQ+uevX42ETdk0JoMVEjNkSOCT7WgERh7C8/+NyhWYNpE3BN69pxFyJIBcF7CxWz/+4A==", - "dev": true - }, - "array-includes": { - "version": "3.1.4", - "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.4.tgz", - "integrity": "sha512-ZTNSQkmWumEbiHO2GF4GmWxYVTiQyJy2XOTa15sdQSrvKn7l+180egQMqlrMOUMCyLMD7pmyQe4mMDUT6Behrw==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "es-abstract": "^1.19.1", - "get-intrinsic": "^1.1.1", - "is-string": "^1.0.7" - } - }, - "array-union": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", - "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", - "dev": true - }, - "array.prototype.flat": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.2.5.tgz", - "integrity": "sha512-KaYU+S+ndVqyUnignHftkwc58o3uVU1jzczILJ1tN2YaIZpFIKBiP/x/j97E5MVPsaCloPbqWLB/8qCTVvT2qg==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "es-abstract": "^1.19.0" - } - }, - "asap": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", - "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=", - "dev": true - }, - "asn1": { - "version": "0.2.6", - "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz", - "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==", - "dev": true, - "requires": { - "safer-buffer": "~2.1.0" - } - }, - "assert-plus": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", - "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", - "dev": true - }, - "ast-metadata-inferer": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/ast-metadata-inferer/-/ast-metadata-inferer-0.7.0.tgz", - "integrity": "sha512-OkMLzd8xelb3gmnp6ToFvvsHLtS6CbagTkFQvQ+ZYFe3/AIl9iKikNR9G7pY3GfOR/2Xc222hwBjzI7HLkE76Q==", - "dev": true, - "requires": { - "@mdn/browser-compat-data": "^3.3.14" - } - }, - "astral-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", - "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", - "dev": true - }, - "astring": { - "version": "1.7.6", - "resolved": "https://registry.npmjs.org/astring/-/astring-1.7.6.tgz", - "integrity": "sha512-7ofuRb7zx2u7T4OGZTtfkGKAfPKq72XQ7zgpI2b3pR3wdROrDIDmKPtrel7D8S4t+97SGpYTpDR6lq5cNX/DJw==", - "dev": true - }, - "async": { - "version": "3.2.3", - "resolved": "https://registry.npmjs.org/async/-/async-3.2.3.tgz", - "integrity": "sha512-spZRyzKL5l5BZQrr/6m/SqFdBN0q3OCI0f9rjfBzCMBIP4p75P620rR3gTmaksNOhmzgdxcaxdNfMy6anrbM0g==", - "dev": true - }, - "asynckit": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", - "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", - "dev": true - }, - "aws-sign2": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", - "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", - "dev": true - }, - "aws4": { - "version": "1.11.0", - "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.11.0.tgz", - "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==", - "dev": true - }, - "axios": { - "version": "0.21.4", - "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.4.tgz", - "integrity": "sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==", - "dev": true, - "requires": { - "follow-redirects": "^1.14.0" - } - }, - "babel-eslint": { - "version": "10.1.0", - "resolved": "https://registry.npmjs.org/babel-eslint/-/babel-eslint-10.1.0.tgz", - "integrity": "sha512-ifWaTHQ0ce+448CYop8AdrQiBsGrnC+bMgfyKFdi6EsPLTAWG+QfyDeM6OH+FmWnKvEq5NnBMLvlBUPKQZoDSg==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.0.0", - "@babel/parser": "^7.7.0", - "@babel/traverse": "^7.7.0", - "@babel/types": "^7.7.0", - "eslint-visitor-keys": "^1.0.0", - "resolve": "^1.12.0" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", - "dev": true - } - } - }, - "babel-jest": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-27.4.5.tgz", - "integrity": "sha512-3uuUTjXbgtODmSv/DXO9nZfD52IyC2OYTFaXGRzL0kpykzroaquCrD5+lZNafTvZlnNqZHt5pb0M08qVBZnsnA==", - "dev": true, - "requires": { - "@jest/transform": "^27.4.5", - "@jest/types": "^27.4.2", - "@types/babel__core": "^7.1.14", - "babel-plugin-istanbul": "^6.0.0", - "babel-preset-jest": "^27.4.0", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.4", - "slash": "^3.0.0" + "@oclif/command": "^1.8.6", + "@oclif/config": "^1.17.1", + "@oclif/errors": "^1.3.5", + "chalk": "^4.1.0", + "debug": "^4.1.0", + "fs-extra": "^9.0.1", + "http-call": "^5.2.2", + "lodash": "^4.17.21", + "semver": "^7.3.2" }, "dependencies": { "ansi-styles": { @@ -3085,11 +3286,42 @@ "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true + "fs-extra": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-9.1.0.tgz", + "integrity": "sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ==", + "dev": true, + "requires": { + "at-least-node": "^1.0.0", + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6", + "universalify": "^2.0.0" + } + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } }, "supports-color": { "version": "7.2.0", @@ -3099,2552 +3331,11346 @@ "requires": { "has-flag": "^4.0.0" } + }, + "universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true } } }, - "babel-plugin-dynamic-import-node": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz", - "integrity": "sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==", - "dev": true, - "requires": { - "object.assign": "^4.1.0" - } - }, - "babel-plugin-istanbul": { - "version": "6.1.1", - "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", - "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "@oclif/plugin-which": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@oclif/plugin-which/-/plugin-which-1.0.4.tgz", + "integrity": "sha512-xuJVhnUDiKIPIw5i6GiQnB3QZ55ie7ucrI6/P0TQWx+1BEt3xl6tEs/vFqnyFuDOh8fc0ZwYTBWFUWkQfYsv4w==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.0.0", - "@istanbuljs/load-nyc-config": "^1.0.0", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-instrument": "^5.0.4", - "test-exclude": "^6.0.0" + "@oclif/command": "1.8.11", + "@oclif/config": "1.18.2", + "cli-ux": "5.6.6", + "tslib": "^2.0.0" }, "dependencies": { - "istanbul-lib-instrument": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.1.0.tgz", - "integrity": "sha512-czwUz525rkOFDJxfKK6mYfIs9zBKILyrZQxjz3ABhjQXhbhFsSbo1HW/BFcsDnfJYJWA6thRR5/TUY2qs5W99Q==", + "@oclif/command": { + "version": "1.8.11", + "resolved": "https://registry.npmjs.org/@oclif/command/-/command-1.8.11.tgz", + "integrity": "sha512-2fGLMvi6J5+oNxTaZfdWPMWY8oW15rYj0V8yLzmZBAEjfzjLqLIzJE9IlNccN1zwRqRHc1bcISSRDdxJ56IS/Q==", + "dev": true, + "requires": { + "@oclif/config": "^1.18.2", + "@oclif/errors": "^1.3.5", + "@oclif/parser": "^3.8.6", + "@oclif/plugin-help": "3.2.14", + "debug": "^4.1.1", + "semver": "^7.3.2" + } + }, + "@oclif/config": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.2.tgz", + "integrity": "sha512-cE3qfHWv8hGRCP31j7fIS7BfCflm/BNZ2HNqHexH+fDrdF2f1D5S8VmXWLC77ffv3oDvWyvE9AZeR0RfmHCCaA==", + "dev": true, + "requires": { + "@oclif/errors": "^1.3.3", + "@oclif/parser": "^3.8.0", + "debug": "^4.1.1", + "globby": "^11.0.1", + "is-wsl": "^2.1.1", + "tslib": "^2.0.0" + } + }, + "@oclif/plugin-help": { + "version": "3.2.14", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-3.2.14.tgz", + "integrity": "sha512-NP5qmE2YfcW3MmXjcrxiqKe9Hf3G0uK/qNc0zAMYKU4crFyIsWj7dBfQVFZSb28YXGioOOpjMzG1I7VMxKF38Q==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.9", + "@oclif/config": "^1.18.2", + "@oclif/errors": "^1.3.5", + "chalk": "^4.1.2", + "indent-string": "^4.0.0", + "lodash": "^4.17.21", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "widest-line": "^3.1.0", + "wrap-ansi": "^6.2.0" + } + }, + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "cli-ux": { + "version": "5.6.6", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.6.tgz", + "integrity": "sha512-4wUB34zoFklcZV0z5YiOM5IqVMMt9c3TK3QYRK3dqyk3XoRC0ybiWDWHfsMDjkKrzsVTw95rXn9NrzSHbae4pg==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.9", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", + "dev": true + }, + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", "dev": true, "requires": { - "@babel/core": "^7.12.3", - "@babel/parser": "^7.14.7", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-coverage": "^3.2.0", - "semver": "^6.3.0" + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" } } } }, - "babel-plugin-jest-hoist": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-27.4.0.tgz", - "integrity": "sha512-Jcu7qS4OX5kTWBc45Hz7BMmgXuJqRnhatqpUhnzGC3OBYpOmf2tv6jFNwZpwM7wU7MUuv2r9IPS/ZlYOuburVw==", + "@oclif/screen": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/@oclif/screen/-/screen-1.0.4.tgz", + "integrity": "sha512-60CHpq+eqnTxLZQ4PGHYNwUX572hgpMHGPtTWMjdTMsAvlm69lZV/4ly6O3sAYkomo4NggGcomrDpBe34rxUqw==", + "dev": true + }, + "@oclif/test": { + "version": "1.2.9", + "resolved": "https://registry.npmjs.org/@oclif/test/-/test-1.2.9.tgz", + "integrity": "sha512-op+ak0NTyeBKqjLVH1jfPCRGWK5befIoQpCL/xwekjucUEmMfCbUpV1Sa60f9rU8X58HEqrclwWbAH+DtQR6FQ==", "dev": true, "requires": { - "@babel/template": "^7.3.3", - "@babel/types": "^7.3.3", - "@types/babel__core": "^7.0.0", - "@types/babel__traverse": "^7.0.6" + "fancy-test": "^1.4.10" } }, - "babel-preset-current-node-syntax": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.1.tgz", - "integrity": "sha512-M7LQ0bxarkxQoN+vz5aJPsLBn77n8QgTFmo8WK0/44auK2xlCXrYcUxHFxgU7qW5Yzw/CjmLRK2uJzaCd7LvqQ==", + "@prettier/plugin-xml": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@prettier/plugin-xml/-/plugin-xml-2.0.1.tgz", + "integrity": "sha512-zJ2+/J5ujo23oHwaiuo3DEnxnRWSw/52IP7to+XmHiTC/qG17w2ld6FbDfVrsnO2PTPp0HFotPtZFlbrPJnaqA==", "dev": true, "requires": { - "@babel/plugin-syntax-async-generators": "^7.8.4", - "@babel/plugin-syntax-bigint": "^7.8.3", - "@babel/plugin-syntax-class-properties": "^7.8.3", - "@babel/plugin-syntax-import-meta": "^7.8.3", - "@babel/plugin-syntax-json-strings": "^7.8.3", - "@babel/plugin-syntax-logical-assignment-operators": "^7.8.3", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", - "@babel/plugin-syntax-numeric-separator": "^7.8.3", - "@babel/plugin-syntax-object-rest-spread": "^7.8.3", - "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", - "@babel/plugin-syntax-optional-chaining": "^7.8.3", - "@babel/plugin-syntax-top-level-await": "^7.8.3" + "@xml-tools/parser": "^1.0.11", + "prettier": ">=2.4.0" } }, - "babel-preset-jest": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-27.4.0.tgz", - "integrity": "sha512-NK4jGYpnBvNxcGo7/ZpZJr51jCGT+3bwwpVIDY2oNfTxJJldRtB4VAcYdgp1loDE50ODuTu+yBjpMAswv5tlpg==", + "@salesforce/apex-node": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@salesforce/apex-node/-/apex-node-0.11.0.tgz", + "integrity": "sha512-7x3QoTcl+5zKQ/PHglBz/IdJuMO65hbzboS2uTMISrFVzeeB06sDeGcv41IfrXn1dxR7NHJm+J3FE70zCgW/7w==", "dev": true, "requires": { - "babel-plugin-jest-hoist": "^27.4.0", - "babel-preset-current-node-syntax": "^1.0.0" + "@salesforce/core": "^2.35.0", + "faye": "1.4.0" } }, - "balanced-match": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", - "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" - }, - "base64-js": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", - "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", - "dev": true - }, - "base64-url": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/base64-url/-/base64-url-2.3.3.tgz", - "integrity": "sha512-dLMhIsK7OplcDauDH/tZLvK7JmUZK3A7KiQpjNzsBrM6Etw7hzNI1tLEywqJk9NnwkgWuFKSlx/IUO7vF6Mo8Q==", - "dev": true - }, - "bcrypt-pbkdf": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", - "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", + "@salesforce/bunyan": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@salesforce/bunyan/-/bunyan-2.0.0.tgz", + "integrity": "sha512-5hq+HWQSeymuygl3i9ehlQo3XWrlBE+A+QzmpDaoK37op4u9M+SBUbXfOW0IABOQCg+JmfQPocSMV74hRoqU9w==", "dev": true, "requires": { - "tweetnacl": "^0.14.3" + "dayjs": "^1.8.16", + "dayjs-plugin-utc": "^0.1.2", + "dtrace-provider": "~0.6", + "mv": "~2", + "safe-json-stringify": "~1" } }, - "bl": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", - "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "@salesforce/command": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-4.2.2.tgz", + "integrity": "sha512-2QEtPIMaeRyUEnLmPHJ1PhfKDJBupfQS5T4nG8rXpK2yOznBu48aPWaWCYErrxyC0bPa5eoFACeyPWz1k9QMog==", "dev": true, "requires": { - "buffer": "^5.5.0", - "inherits": "^2.0.4", - "readable-stream": "^3.4.0" + "@oclif/command": "^1.8.1", + "@oclif/errors": "^1.2.2", + "@oclif/parser": "3.8.6", + "@oclif/plugin-help": "^2.2.0", + "@oclif/test": "^1.2.4", + "@salesforce/core": "^2.35.0", + "@salesforce/kit": "^1.5.17", + "@salesforce/ts-types": "^1.5.20", + "chalk": "^2.4.2", + "cli-ux": "^4.9.3" }, "dependencies": { - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "@oclif/parser": { + "version": "3.8.6", + "resolved": "https://registry.npmjs.org/@oclif/parser/-/parser-3.8.6.tgz", + "integrity": "sha512-tXb0NKgSgNxmf6baN6naK+CCwOueaFk93FG9u202U7mTBHUKsioOUlw1SG/iPi9aJM3WE4pHLXmty59pci0OEw==", "dev": true, "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" + "@oclif/errors": "^1.2.2", + "@oclif/linewrap": "^1.0.0", + "chalk": "^4.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + } + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" } } } }, - "bluebird": { - "version": "3.7.2", - "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", - "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", - "dev": true - }, - "brace-expansion": { - "version": "1.1.11", - "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", - "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "braces": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", - "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, - "requires": { - "fill-range": "^7.0.1" - } - }, - "browser-process-hrtime": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz", - "integrity": "sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==", - "dev": true - }, - "browserslist": { - "version": "4.19.1", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.19.1.tgz", - "integrity": "sha512-u2tbbG5PdKRTUoctO3NBD8FQ5HdPh1ZXPHzp1rwaa5jTc+RV9/+RlWiAIKmjRPQF+xbGM9Kklj5bZQFa2s/38A==", - "dev": true, - "requires": { - "caniuse-lite": "^1.0.30001286", - "electron-to-chromium": "^1.4.17", - "escalade": "^3.1.1", - "node-releases": "^2.0.1", - "picocolors": "^1.0.0" - } - }, - "bser": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", - "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", - "dev": true, - "requires": { - "node-int64": "^0.4.0" - } - }, - "buffer": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", - "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", - "dev": true, - "requires": { - "base64-js": "^1.3.1", - "ieee754": "^1.1.13" - } - }, - "buffer-crc32": { - "version": "0.2.13", - "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", - "integrity": "sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI=", - "dev": true - }, - "buffer-equal-constant-time": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", - "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=", - "dev": true - }, - "buffer-from": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", - "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", - "dev": true - }, - "cache-point": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/cache-point/-/cache-point-2.0.0.tgz", - "integrity": "sha512-4gkeHlFpSKgm3vm2gJN5sPqfmijYRFYCQ6tv5cLw0xVmT6r1z1vd4FNnpuOREco3cBs1G709sZ72LdgddKvL5w==", + "@salesforce/core": { + "version": "2.36.0", + "resolved": "https://registry.npmjs.org/@salesforce/core/-/core-2.36.0.tgz", + "integrity": "sha512-VsKt7SXArxrOelaJl5Ez21Pmtdp2eU6qimqZ5clxnNaZDZKOySDPYnKHu7AYCt1LUNFN9cfsX8K5yOHxS+wT+w==", "dev": true, "requires": { - "array-back": "^4.0.1", - "fs-then-native": "^2.0.0", - "mkdirp2": "^1.0.4" + "@salesforce/bunyan": "^2.0.0", + "@salesforce/kit": "^1.5.17", + "@salesforce/schemas": "^1.0.1", + "@salesforce/ts-types": "^1.5.20", + "@types/graceful-fs": "^4.1.5", + "@types/jsforce": "^1.9.41", + "@types/mkdirp": "^1.0.1", + "archiver": "^5.3.0", + "debug": "^3.1.0", + "faye": "^1.4.0", + "graceful-fs": "^4.2.4", + "js2xmlparser": "^4.0.1", + "jsen": "0.6.6", + "jsforce": "^1.11.0", + "jsonwebtoken": "8.5.0", + "mkdirp": "1.0.4", + "semver": "^7.3.5", + "ts-retry-promise": "^0.6.0" }, "dependencies": { - "array-back": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-4.0.2.tgz", - "integrity": "sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==", + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } } } }, - "call-bind": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", - "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "@salesforce/eslint-config-lwc": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/@salesforce/eslint-config-lwc/-/eslint-config-lwc-3.3.0.tgz", + "integrity": "sha512-eisGB/HsAlkzrClrHdcixmIP4b/0sfD+U6WMwIvMjsQ10WkIemuPMEr15Jr3nDQI61iQujlpkmEl5Iqp7ZbZRA==", "dev": true, "requires": { - "function-bind": "^1.1.1", - "get-intrinsic": "^1.0.2" + "@babel/core": "~7.17.8", + "@babel/eslint-parser": "~7.17.0", + "eslint-restricted-globals": "~0.2.0", + "semver": "^7.3.5" + }, + "dependencies": { + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } } }, - "callsites": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", - "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", - "dev": true - }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, - "caniuse-lite": { - "version": "1.0.30001295", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001295.tgz", - "integrity": "sha512-lSP16vcyC0FEy0R4ECc9duSPoKoZy+YkpGkue9G4D81OfPnliopaZrU10+qtPdT8PbGXad/PNx43TIQrOmJZSQ==", - "dev": true - }, - "cardinal": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/cardinal/-/cardinal-2.1.1.tgz", - "integrity": "sha1-fMEFXYItISlU0HsIXeolHMe8VQU=", + "@salesforce/eslint-plugin-aura": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@salesforce/eslint-plugin-aura/-/eslint-plugin-aura-2.1.0.tgz", + "integrity": "sha512-7HfwSBKTHQZQboLoEhkBY7bYR9wTaT+G5jHXGlq8y31hEnNhJXRZ+RERDEwDm1jYa2SV9lE8nMNr0/8EKIGjlQ==", "dev": true, "requires": { - "ansicolors": "~0.3.2", - "redeyed": "~2.1.0" + "eslint-plugin-compat": "^4.0.2" } }, - "caseless": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", - "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", + "@salesforce/eslint-plugin-lightning": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@salesforce/eslint-plugin-lightning/-/eslint-plugin-lightning-1.0.0.tgz", + "integrity": "sha512-zk0PKXAcHKHepAG2EOSWlkOTxQM0Aw1CT6+MUxJcM42fCDwH/yPPpGkG3CWtRfmVViODGOwU9ywU2wlkAYcvUQ==", "dev": true }, - "catharsis": { - "version": "0.9.0", - "resolved": "https://registry.npmjs.org/catharsis/-/catharsis-0.9.0.tgz", - "integrity": "sha512-prMTQVpcns/tzFgFVkVp6ak6RykZyWb3gu8ckUpd6YkTlacOd3DXGJjIpD4Q6zJirizvaiAjSSHlOsA+6sNh2A==", + "@salesforce/kit": { + "version": "1.5.38", + "resolved": "https://registry.npmjs.org/@salesforce/kit/-/kit-1.5.38.tgz", + "integrity": "sha512-tAz5oWTlHW4VkQqWilaHihuiFf/opzbhfL6JK1TPtY8CyKCwCMPchoWafe+pCKWOSFTu+WzT/er06zVYC4ITrw==", "dev": true, "requires": { - "lodash": "^4.17.15" + "@salesforce/ts-types": "^1.5.20", + "shx": "^0.3.3", + "tslib": "^2.2.0" } }, - "chalk": { - "version": "2.4.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", - "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "@salesforce/lazy-require": { + "version": "0.4.21", + "resolved": "https://registry.npmjs.org/@salesforce/lazy-require/-/lazy-require-0.4.21.tgz", + "integrity": "sha512-JOfqvHXXhTRjJre84N8vB7u8s3g+8DzhDSHimCuMIn2uv7SQil5TXmvAu8eld7D5L9yzs1w25kGVeCHdUK/lAw==", "dev": true, "requires": { - "ansi-styles": "^3.2.1", - "escape-string-regexp": "^1.0.5", - "supports-color": "^5.3.0" + "debug": "^4.3.1", + "tslib": "^2.2.0" } }, - "char-regex": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", - "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", - "dev": true - }, - "chardet": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", - "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", - "dev": true - }, - "chevrotain": { - "version": "7.1.1", - "resolved": "https://registry.npmjs.org/chevrotain/-/chevrotain-7.1.1.tgz", - "integrity": "sha512-wy3mC1x4ye+O+QkEinVJkPf5u2vsrDIYW9G7ZuwFl6v/Yu0LwUuT2POsb+NUWApebyxfkQq6+yDfRExbnI5rcw==", + "@salesforce/plugin-alias": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-alias/-/plugin-alias-2.0.0.tgz", + "integrity": "sha512-v4nS3dw7qxbWqYja7XL6xPZuJPj7hiy9kuam4bxOyMgqGQBGjeeMbdjJpNmv2MaeAC/+QoZWSB6MUjnP6GTAEw==", "dev": true, "requires": { - "regexp-to-ast": "0.5.0" - } - }, - "chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", - "dev": true - }, - "ci-info": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.3.0.tgz", - "integrity": "sha512-riT/3vI5YpVH6/qomlDnJow6TBee2PBKSEpx3O32EGPYbWGIRsIlGRms3Sm74wYE1JMo8RnO04Hb12+v1J5ICw==", - "dev": true - }, - "cjs-module-lexer": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.2.2.tgz", - "integrity": "sha512-cOU9usZw8/dXIXKtwa8pM0OTJQuJkxMN6w30csNRUerHfeQ5R6U3kkU/FtJeIf3M202OHfY2U8ccInBG7/xogA==", - "dev": true - }, - "clean-stack": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-3.0.1.tgz", - "integrity": "sha512-lR9wNiMRcVQjSB3a7xXGLuz4cr4wJuuXlaAEbRutGowQTmlp7R72/DOgN21e8jdwblMWl9UOJMJXarX94pzKdg==", - "dev": true, - "requires": { - "escape-string-regexp": "4.0.0" - }, - "dependencies": { - "escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true - } - } - }, - "cli-cursor": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", - "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", - "dev": true, - "requires": { - "restore-cursor": "^3.1.0" - } - }, - "cli-truncate": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/cli-truncate/-/cli-truncate-3.1.0.tgz", - "integrity": "sha512-wfOBkjXteqSnI59oPcJkcPl/ZmwvMMOj340qUIY1SKZCv0B9Cf4D4fAucRkIKQmsIuYK3x1rrgU7MeGRruiuiA==", - "dev": true, - "requires": { - "slice-ansi": "^5.0.0", - "string-width": "^5.0.0" + "@oclif/core": "^1.5.1", + "@salesforce/command": "^5.0.0", + "@salesforce/core": "^3.7.8", + "chalk": "^4.1.2", + "tslib": "^2" }, "dependencies": { - "ansi-regex": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz", - "integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==", - "dev": true - }, - "ansi-styles": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.1.0.tgz", - "integrity": "sha512-VbqNsoz55SYGczauuup0MFUyXNQviSpFTj1RQtFzmQLk18qbVSpTFFGMT293rmDaQuKCT6InmbuEyUne4mTuxQ==", - "dev": true + "@oclif/plugin-help": { + "version": "5.1.12", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-5.1.12.tgz", + "integrity": "sha512-HvH/RubJxqCinP0vUWQLTOboT+SfjfL8h40s+PymkWaldIcXlpoRaJX50vz+SjZIs7uewZwEk8fzLqpF/BWXlg==", + "dev": true, + "requires": { + "@oclif/core": "^1.3.6" + } }, - "emoji-regex": { - "version": "9.2.2", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", - "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", - "dev": true + "@oclif/test": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@oclif/test/-/test-2.1.0.tgz", + "integrity": "sha512-o+JTv3k28aMUxywJUlJY1/DORLqumoZFRII492phOmtXM16rD6Luy3z1qinT4BvEtPj2BzOPd2whr/VdYszaYw==", + "dev": true, + "requires": { + "@oclif/core": "^1.3.1", + "fancy-test": "^2.0.0" + } }, - "is-fullwidth-code-point": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-4.0.0.tgz", - "integrity": "sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ==", - "dev": true + "@salesforce/command": { + "version": "5.0.5", + "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-5.0.5.tgz", + "integrity": "sha512-zutR54SbdBZNzb2UVKdfe7/3sPMi5v8gPaSWvTSRjaOmRI476EE5xosKc0X1Svm39QKobho04Rkqn8r+r78lgA==", + "dev": true, + "requires": { + "@oclif/core": "^1.5.2", + "@oclif/plugin-help": "^5.1.11", + "@oclif/test": "^2.1.0", + "@salesforce/core": "^3.7.9", + "@salesforce/kit": "^1.5.34", + "@salesforce/ts-types": "^1.5.20", + "chalk": "^2.4.2" + }, + "dependencies": { + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + } + } }, - "slice-ansi": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-5.0.0.tgz", - "integrity": "sha512-FC+lgizVPfie0kkhqUScwRu1O/lF6NOgJmlCgK+/LYxDCTk8sGelYaHDhFcDN+Sn3Cv+3VSa4Byeo+IMCzpMgQ==", + "@salesforce/core": { + "version": "3.12.1", + "resolved": "https://registry.npmjs.org/@salesforce/core/-/core-3.12.1.tgz", + "integrity": "sha512-j7WvlPwhTLBMBdnF/6Og3oqcM72tGqK99UHK/UqJpGaX0HcadnxMyL3uOMi8aAautvg4BnpfGjesn0pinqKVXw==", "dev": true, "requires": { - "ansi-styles": "^6.0.0", - "is-fullwidth-code-point": "^4.0.0" + "@salesforce/bunyan": "^2.0.0", + "@salesforce/kit": "^1.5.34", + "@salesforce/schemas": "^1.1.0", + "@salesforce/ts-types": "^1.5.20", + "@types/graceful-fs": "^4.1.5", + "@types/mkdirp": "^1.0.2", + "@types/semver": "^7.3.9", + "archiver": "^5.3.0", + "change-case": "^4.1.2", + "debug": "^3.2.7", + "faye": "^1.4.0", + "form-data": "^4.0.0", + "graceful-fs": "^4.2.9", + "js2xmlparser": "^4.0.1", + "jsen": "0.6.6", + "jsforce": "2.0.0-beta.7", + "jsonwebtoken": "8.5.1", + "mkdirp": "1.0.4", + "ts-retry-promise": "^0.6.0" } }, - "string-width": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.0.tgz", - "integrity": "sha512-7x54QnN21P+XL/v8SuNKvfgsUre6PXpN7mc77N3HlZv+f1SBRGmjxtOud2Z6FZ8DmdkD/IdjCaf9XXbnqmTZGQ==", + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, "requires": { - "eastasianwidth": "^0.2.0", - "emoji-regex": "^9.2.2", - "strip-ansi": "^7.0.1" + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } } }, - "strip-ansi": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.0.1.tgz", - "integrity": "sha512-cXNxvT8dFNRVfhVME3JAe98mkXDYN2O1l7jmcwMnOslDeESg1rF/OZMtK0nRAhiari1unG5cD4jG3rapUAkLbw==", + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", "dev": true, "requires": { - "ansi-regex": "^6.0.1" + "color-name": "~1.1.4" } - } - } - }, - "cli-ux": { - "version": "4.9.3", - "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-4.9.3.tgz", - "integrity": "sha512-/1owvF0SZ5Gn54cgrikJ0QskgTzeg30HGjkmjFoaHDJzAqFpuX1DBpFR8aLvsE1J5s9MgeYRENQK4BFwOag5VA==", - "dev": true, - "requires": { - "@oclif/errors": "^1.2.2", - "@oclif/linewrap": "^1.0.0", - "@oclif/screen": "^1.0.3", - "ansi-escapes": "^3.1.0", - "ansi-styles": "^3.2.1", - "cardinal": "^2.1.1", - "chalk": "^2.4.1", - "clean-stack": "^2.0.0", - "extract-stack": "^1.0.0", - "fs-extra": "^7.0.0", - "hyperlinker": "^1.0.0", - "indent-string": "^3.2.0", - "is-wsl": "^1.1.0", - "lodash": "^4.17.11", - "password-prompt": "^1.0.7", - "semver": "^5.6.0", - "strip-ansi": "^5.0.0", - "supports-color": "^5.5.0", - "supports-hyperlinks": "^1.0.1", - "treeify": "^1.1.0", - "tslib": "^1.9.3" - }, - "dependencies": { - "ansi-escapes": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", - "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "commander": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz", + "integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==", "dev": true }, - "clean-stack": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", - "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", + "csv-stringify": { + "version": "5.6.5", + "resolved": "https://registry.npmjs.org/csv-stringify/-/csv-stringify-5.6.5.tgz", + "integrity": "sha512-PjiQ659aQ+fUTQqSrd1XEDnOr52jh30RBurfzkscaE2tPaFsDH5wOAHJiw8XAHphRknCwMUE9KRayc4K/NbO8A==", "dev": true }, - "fs-extra": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", - "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==", + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", "dev": true, "requires": { - "graceful-fs": "^4.1.2", - "jsonfile": "^4.0.0", - "universalify": "^0.1.0" + "ms": "^2.1.1" } }, - "has-flag": { + "fancy-test": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-2.0.0.tgz", - "integrity": "sha1-6CB68cx7MNRGzHC3NLXovhj4jVE=", - "dev": true + "resolved": "https://registry.npmjs.org/fancy-test/-/fancy-test-2.0.0.tgz", + "integrity": "sha512-SFb2D/VX9SV+wNYXO1IIh1wyxUC1GS0mYCFJOWD1ia7MPj9yE2G8jaPkw4t/pg0Sj7/YJP56OzMY4nAuJSOugQ==", + "dev": true, + "requires": { + "@types/chai": "*", + "@types/lodash": "*", + "@types/node": "*", + "@types/sinon": "*", + "lodash": "^4.17.13", + "mock-stdin": "^1.0.0", + "nock": "^13.0.0", + "stdout-stderr": "^0.1.9" + } }, - "indent-string": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-3.2.0.tgz", - "integrity": "sha1-Sl/W0nzDMvN+VBmlBNu4NxBckok=", - "dev": true + "form-data": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", + "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "dev": true, + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + } }, - "is-wsl": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", - "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", - "dev": true + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "jsforce": { + "version": "2.0.0-beta.7", + "resolved": "https://registry.npmjs.org/jsforce/-/jsforce-2.0.0-beta.7.tgz", + "integrity": "sha512-aiJRbf6v0eQSJLpAg4aEB/yXsQwV9WM3ewT2v/WTLmeeZQ4ZtwlcJhQCTwW4tKX/S8U+t5nL+Iluz8jFSZFqnA==", "dev": true, "requires": { - "ansi-regex": "^4.1.0" + "@babel/runtime": "^7.12.5", + "@babel/runtime-corejs3": "^7.12.5", + "@types/node": "^12.19.9", + "abort-controller": "^3.0.0", + "base64url": "^3.0.1", + "commander": "^4.0.1", + "core-js": "^3.6.4", + "csv-parse": "^4.8.2", + "csv-stringify": "^5.3.4", + "faye": "^1.4.0", + "fs-extra": "^8.1.0", + "https-proxy-agent": "^5.0.0", + "inquirer": "^7.0.0", + "multistream": "^3.1.0", + "node-fetch": "^2.6.1", + "open": "^7.0.0", + "regenerator-runtime": "^0.13.3", + "strip-ansi": "^6.0.0", + "xml2js": "^0.4.22" + }, + "dependencies": { + "@types/node": { + "version": "12.20.47", + "resolved": "https://registry.npmjs.org/@types/node/-/node-12.20.47.tgz", + "integrity": "sha512-BzcaRsnFuznzOItW1WpQrDHM7plAa7GIDMZ6b5pnMbkqEtM/6WCOhvZar39oeMQP79gwvFUWjjptE7/KGcNqFg==", + "dev": true + } } }, - "supports-hyperlinks": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-1.0.1.tgz", - "integrity": "sha512-HHi5kVSefKaJkGYXbDuKbUGRVxqnWGn3J2e39CYcNJEfWciGq2zYtOhXLTlvrOZW1QU7VX67w7fMmWafHX9Pfw==", + "jsonwebtoken": { + "version": "8.5.1", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.1.tgz", + "integrity": "sha512-XjwVfRS6jTMsqYs0EsuJ4LGxXV14zQybNd4L2r0UvbVnSF9Af8x7p5MzbJ90Ioz/9TI41/hTCvznF/loiSzn8w==", "dev": true, "requires": { - "has-flag": "^2.0.0", - "supports-color": "^5.0.0" + "jws": "^3.2.2", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^5.6.0" } }, - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", "dev": true - } - } - }, - "cli-width": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz", - "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", - "dev": true - }, - "cliui": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", - "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", - "dev": true, - "requires": { - "string-width": "^4.2.0", - "strip-ansi": "^6.0.0", - "wrap-ansi": "^7.0.0" - } - }, - "co": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", - "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=", - "dev": true - }, - "co-prompt": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/co-prompt/-/co-prompt-1.0.0.tgz", - "integrity": "sha1-+zcOntrEhXayenMv5dfyHZ/G5vY=", - "dev": true, - "requires": { - "keypress": "~0.2.1" - } - }, - "coffeescript": { - "version": "1.12.7", - "resolved": "https://registry.npmjs.org/coffeescript/-/coffeescript-1.12.7.tgz", - "integrity": "sha512-pLXHFxQMPklVoEekowk8b3erNynC+DVJzChxS/LCBBgR6/8AJkHivkm//zbowcfc7BTCAjryuhx6gPqPRfsFoA==", - "dev": true - }, - "collect-all": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/collect-all/-/collect-all-1.0.4.tgz", - "integrity": "sha512-RKZhRwJtJEP5FWul+gkSMEnaK6H3AGPTTWOiRimCcs+rc/OmQE3Yhy1Q7A7KsdkG3ZXVdZq68Y6ONSdvkeEcKA==", - "dev": true, - "requires": { - "stream-connect": "^1.0.2", - "stream-via": "^1.0.4" - } - }, - "collect-v8-coverage": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.1.tgz", - "integrity": "sha512-iBPtljfCNcTKNAto0KEtDfZ3qzjJvqE3aTGZsbhjSBlorqpXJlaWWtPO35D+ZImoC3KWejX64o+yPGxhWSTzfg==", - "dev": true - }, - "color-convert": { - "version": "1.9.3", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", - "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", - "dev": true, - "requires": { - "color-name": "1.1.3" - } - }, - "color-name": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", - "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", - "dev": true - }, - "colorette": { - "version": "2.0.16", - "resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.16.tgz", - "integrity": "sha512-hUewv7oMjCp+wkBv5Rm0v87eJhq4woh5rSR+42YSQJKecCqgIqNkZ6lAlQms/BwHPJA5NKMRlpxPRv0n8HQW6g==", - "dev": true - }, - "colors": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", - "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==", - "dev": true - }, - "combined-stream": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", - "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", - "dev": true, - "requires": { - "delayed-stream": "~1.0.0" - } - }, - "command-line-args": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/command-line-args/-/command-line-args-5.2.0.tgz", - "integrity": "sha512-4zqtU1hYsSJzcJBOcNZIbW5Fbk9BkjCp1pZVhQKoRaWL5J7N4XphDLwo8aWwdQpTugxwu+jf9u2ZhkXiqp5Z6A==", - "dev": true, - "requires": { - "array-back": "^3.1.0", - "find-replace": "^3.0.0", - "lodash.camelcase": "^4.3.0", - "typical": "^4.0.0" - }, - "dependencies": { - "array-back": { + }, + "multistream": { "version": "3.1.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz", - "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==", - "dev": true + "resolved": "https://registry.npmjs.org/multistream/-/multistream-3.1.0.tgz", + "integrity": "sha512-zBgD3kn8izQAN/TaL1PCMv15vYpf+Vcrsfub06njuYVYlzUldzpopTlrEZ53pZVEbfn3Shtv7vRFoOv6LOV87Q==", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "readable-stream": "^3.4.0" + } }, - "typical": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/typical/-/typical-4.0.0.tgz", - "integrity": "sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw==", + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", "dev": true } } }, - "command-line-tool": { - "version": "0.8.0", - "resolved": "https://registry.npmjs.org/command-line-tool/-/command-line-tool-0.8.0.tgz", - "integrity": "sha512-Xw18HVx/QzQV3Sc5k1vy3kgtOeGmsKIqwtFFoyjI4bbcpSgnw2CWVULvtakyw4s6fhyAdI6soQQhXc2OzJy62g==", + "@salesforce/plugin-apex": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-apex/-/plugin-apex-0.11.0.tgz", + "integrity": "sha512-8GDbrT1jIhDXgQhbS26vihSKWxng5OMjjzzEoY6WKEwXtBwnsR4WRm232Z7LbtTy6p7aw7K1wTPZYj6vDMiduQ==", "dev": true, "requires": { - "ansi-escape-sequences": "^4.0.0", - "array-back": "^2.0.0", - "command-line-args": "^5.0.0", - "command-line-usage": "^4.1.0", - "typical": "^2.6.1" + "@oclif/command": "^1", + "@oclif/config": "^1", + "@oclif/errors": "^1", + "@salesforce/apex-node": "0.11.0", + "@salesforce/command": "4.2.0", + "@salesforce/core": "^2.35.0", + "chalk": "^4.1.0", + "tslib": "^1" }, "dependencies": { - "array-back": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-2.0.0.tgz", - "integrity": "sha512-eJv4pLLufP3g5kcZry0j6WXpIbzYw9GUB4mVJZno9wfwiBxbizTnHCw3VJb07cBihbFX48Y7oSrW9y+gt4glyw==", + "@oclif/parser": { + "version": "3.8.6", + "resolved": "https://registry.npmjs.org/@oclif/parser/-/parser-3.8.6.tgz", + "integrity": "sha512-tXb0NKgSgNxmf6baN6naK+CCwOueaFk93FG9u202U7mTBHUKsioOUlw1SG/iPi9aJM3WE4pHLXmty59pci0OEw==", "dev": true, "requires": { - "typical": "^2.6.1" + "@oclif/errors": "^1.2.2", + "@oclif/linewrap": "^1.0.0", + "chalk": "^4.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "tslib": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.3.1.tgz", + "integrity": "sha512-77EbyPPpMz+FRFRuAFlWMtmgUWGe9UOG2Z25NqCwiIjRhOf5iKGuzSe5P2w1laq+FkRy4p+PCuVkJSGkzTEKVw==", + "dev": true + } + } + }, + "@salesforce/command": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-4.2.0.tgz", + "integrity": "sha512-KxvzBHqJHLR5WMP/Gh4TpSjCSfIveZEVRA8+zS+ZFXXLILuZzCPi/kHwZnKwOedO3EyjC8HFnFDgKt62vqgw8w==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.1", + "@oclif/errors": "^1.2.2", + "@oclif/parser": "3.8.6", + "@oclif/plugin-help": "^2.2.0", + "@oclif/test": "^1.2.4", + "@salesforce/core": "^2.31.0", + "@salesforce/kit": "^1.5.17", + "@salesforce/ts-types": "^1.5.20", + "chalk": "^2.4.2", + "cli-ux": "^4.9.3" + }, + "dependencies": { + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", + "dev": true + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "dev": true + }, + "supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "requires": { + "has-flag": "^3.0.0" + } + } + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true } } }, - "command-line-usage": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/command-line-usage/-/command-line-usage-4.1.0.tgz", - "integrity": "sha512-MxS8Ad995KpdAC0Jopo/ovGIroV/m0KHwzKfXxKag6FHOkGsH8/lv5yjgablcRxCJJC0oJeUMuO/gmaq+Wq46g==", + "@salesforce/plugin-auth": { + "version": "1.8.2", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-auth/-/plugin-auth-1.8.2.tgz", + "integrity": "sha512-ROHxuEHsMv5uh1LA6BWqvYEx2ajqYHzcX6FDHP0TisEARO5yIq+2Mvhl2ob2jENsDjHyXurA2U1m/MVimu7qbw==", "dev": true, "requires": { - "ansi-escape-sequences": "^4.0.0", - "array-back": "^2.0.0", - "table-layout": "^0.4.2", - "typical": "^2.6.1" + "@oclif/config": "^1.18.1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.31.0", + "@salesforce/kit": "^1.5.17", + "chalk": "^4.1.2", + "open": "^8.2.1", + "tslib": "^2" }, "dependencies": { - "array-back": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-2.0.0.tgz", - "integrity": "sha512-eJv4pLLufP3g5kcZry0j6WXpIbzYw9GUB4mVJZno9wfwiBxbizTnHCw3VJb07cBihbFX48Y7oSrW9y+gt4glyw==", + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", "dev": true, "requires": { - "typical": "^2.6.1" + "color-convert": "^2.0.1" } - } - } - }, - "commander": { - "version": "2.20.3", - "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", - "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", - "dev": true - }, - "common-sequence": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/common-sequence/-/common-sequence-2.0.2.tgz", - "integrity": "sha512-jAg09gkdkrDO9EWTdXfv80WWH3yeZl5oT69fGfedBNS9pXUKYInVJ1bJ+/ht2+Moeei48TmSbQDYMc8EOx9G0g==", - "dev": true - }, - "compress-commons": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/compress-commons/-/compress-commons-4.1.1.tgz", - "integrity": "sha512-QLdDLCKNV2dtoTorqgxngQCMA+gWXkM/Nwu7FpeBhk/RdkzimqC3jueb/FDmaZeXh+uby1jkBqE3xArsLBE5wQ==", - "dev": true, - "requires": { - "buffer-crc32": "^0.2.13", - "crc32-stream": "^4.0.2", - "normalize-path": "^3.0.0", - "readable-stream": "^3.6.0" - }, - "dependencies": { - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", "dev": true, "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" } - } - } - }, - "concat-map": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" - }, - "config-master": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/config-master/-/config-master-3.1.0.tgz", - "integrity": "sha1-ZnZjWQUFooO/JqSE1oSJ10xUhdo=", - "dev": true, - "requires": { - "walk-back": "^2.0.1" - }, - "dependencies": { - "walk-back": { + }, + "color-convert": { "version": "2.0.1", - "resolved": "https://registry.npmjs.org/walk-back/-/walk-back-2.0.1.tgz", - "integrity": "sha1-VU4qnYdPrEeoywBr9EwvDEmYoKQ=", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true + }, + "open": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/open/-/open-8.4.0.tgz", + "integrity": "sha512-XgFPPM+B28FtCCgSb9I+s9szOC1vZRSwgWsRUA5ylIxRTgKozqjOCrVOqGsYABPYK5qnfqClxZTFBa8PKt2v6Q==", + "dev": true, + "requires": { + "define-lazy-prop": "^2.0.0", + "is-docker": "^2.1.1", + "is-wsl": "^2.2.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } } } }, - "contains-path": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", - "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=", - "dev": true - }, - "convert-source-map": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.8.0.tgz", - "integrity": "sha512-+OQdjP49zViI/6i7nIJpA8rAl4sV/JdPfU9nZs3VqOwGIgizICvuN2ru6fMd+4llL0tar18UYJXfZ/TWtmhUjA==", + "@salesforce/plugin-community": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-community/-/plugin-community-1.1.4.tgz", + "integrity": "sha512-pmbnfOMe2G/vfZuoNWaoH3hZMQqcW2ge3CBYbmUe4wZRT+vjP8ZcTtJLZOMSYvLnFUycY5KRx3gnZlQLo3dsYQ==", "dev": true, "requires": { - "safe-buffer": "~5.1.1" + "@oclif/config": "^1", + "@salesforce/command": "^4.2.2", + "@salesforce/core": "^2.35.1", + "tslib": "^2" } }, - "core-js": { - "version": "3.20.2", - "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.20.2.tgz", - "integrity": "sha512-nuqhq11DcOAbFBV4zCbKeGbKQsUDRqTX0oqx7AttUBuqe3h20ixsE039QHelbL6P4h+9kytVqyEtyZ6gsiwEYw==", - "dev": true - }, - "core-util-is": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", - "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", - "dev": true - }, - "crc-32": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.1.tgz", - "integrity": "sha512-Dn/xm/1vFFgs3nfrpEVScHoIslO9NZRITWGz/1E/St6u4xw99vfZzVkW0OSnzx2h9egej9xwMCEut6sqwokM/w==", + "@salesforce/plugin-config": { + "version": "1.3.24", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-config/-/plugin-config-1.3.24.tgz", + "integrity": "sha512-zOxQL8ZYIrNhVLIPXLl934YUFHzyyCkMuBxY3nWkOZ6m1otxUEyXAAWyqIVR1xVMIXymAlq/Xs2ZSjLVfTjZIA==", "dev": true, "requires": { - "exit-on-epipe": "~1.0.1", - "printj": "~1.3.1" + "@oclif/config": "^1.18.1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.31.0", + "tslib": "^2" } }, - "crc32-stream": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/crc32-stream/-/crc32-stream-4.0.2.tgz", - "integrity": "sha512-DxFZ/Hk473b/muq1VJ///PMNLj0ZMnzye9thBpmjpJKCc5eMgB95aK8zCGrGfQ90cWo561Te6HK9D+j4KPdM6w==", + "@salesforce/plugin-custom-metadata": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-custom-metadata/-/plugin-custom-metadata-1.0.12.tgz", + "integrity": "sha512-f/I8rxmIuOMQtJ4SbWIvf+ui4Nak/of0jwxS90mNcG1Fvtu8vikYvaXzGA6sdiOVkMFrDhHHJkf0D416Bu/zBQ==", "dev": true, "requires": { - "crc-32": "^1.2.0", - "readable-stream": "^3.4.0" + "@oclif/command": "^1.6.1", + "@oclif/config": "^1.15.1", + "@oclif/errors": "^1.2.2", + "@salesforce/command": "^3.0.0", + "@salesforce/ts-types": "^1.2.2", + "tslib": "^2.0.0" }, "dependencies": { - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "@salesforce/command": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-3.1.3.tgz", + "integrity": "sha512-Yg9lhl3ghwPN7WwqFmgfWIn6i7vz43WTpEsYsChz80bKORlVbDvhwPZQUj0XTv3DKDnPZVU8FFIDsrLSOa9G1A==", "dev": true, "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" + "@oclif/command": "^1.5.17", + "@oclif/errors": "^1.2.2", + "@oclif/parser": "^3.8.3", + "@oclif/plugin-help": "^2.2.0", + "@oclif/test": "^1.2.4", + "@salesforce/core": "^2.23.4", + "@salesforce/kit": "^1.2.2", + "@salesforce/ts-types": "^1.2.0", + "chalk": "^2.4.2", + "cli-ux": "^4.9.3" } } } }, - "cross-spawn": { - "version": "7.0.3", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", - "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", - "dev": true, - "requires": { - "path-key": "^3.1.0", - "shebang-command": "^2.0.0", - "which": "^2.0.1" - } - }, - "csprng": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/csprng/-/csprng-0.1.2.tgz", - "integrity": "sha1-S8aPEvo2jSUqWYQcusqXSxirReI=", - "dev": true, - "requires": { - "sequin": "*" - } - }, - "cssesc": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", - "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", - "dev": true - }, - "cssom": { - "version": "0.4.4", - "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.4.4.tgz", - "integrity": "sha512-p3pvU7r1MyyqbTk+WbNJIgJjG2VmTIaB10rI93LzVPrmDJKkzKYMtxxyAvQXR/NS6otuzveI7+7BBq3SjBS2mw==", - "dev": true - }, - "cssstyle": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz", - "integrity": "sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A==", + "@salesforce/plugin-data": { + "version": "0.6.12", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-data/-/plugin-data-0.6.12.tgz", + "integrity": "sha512-gyGaN7p6SlZwBEatGrC/CobUqqRtnA+ZR/W14unWwoKNi1asoKoM4I/ddOlktE4sBvBJzR0AwBtYp83iL1MTMA==", "dev": true, "requires": { - "cssom": "~0.3.6" + "@oclif/config": "^1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.35.3", + "@salesforce/ts-types": "^1.5.20", + "chalk": "^4.1.0", + "csv-parse": "^4.16.3", + "tslib": "^2" }, "dependencies": { - "cssom": { - "version": "0.3.8", - "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", - "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==", + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } } } }, - "csv-parse": { - "version": "4.16.3", - "resolved": "https://registry.npmjs.org/csv-parse/-/csv-parse-4.16.3.tgz", - "integrity": "sha512-cO1I/zmz4w2dcKHVvpCr7JVRu8/FymG5OEpmvsZYlccYolPBLoVGKUHgNoc4ZGkFeFlWGEDmMyBM+TTqRdW/wg==", - "dev": true - }, - "csv-stringify": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/csv-stringify/-/csv-stringify-1.1.2.tgz", - "integrity": "sha1-d6QVJlgbzjOA8SsA18W7rHDIK1g=", - "dev": true, - "requires": { - "lodash.get": "~4.4.2" - } - }, - "dashdash": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", - "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0" - } - }, - "data-urls": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", - "integrity": "sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ==", - "dev": true, - "requires": { - "abab": "^2.0.3", - "whatwg-mimetype": "^2.3.0", - "whatwg-url": "^8.0.0" - } - }, - "dayjs": { - "version": "1.10.7", - "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.10.7.tgz", - "integrity": "sha512-P6twpd70BcPK34K26uJ1KT3wlhpuOAPoMwJzpsIWUxHZ7wpmbdZL/hQqBDfz7hGurYSa5PhzdhDHtt319hL3ig==", - "dev": true - }, - "dayjs-plugin-utc": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/dayjs-plugin-utc/-/dayjs-plugin-utc-0.1.2.tgz", - "integrity": "sha512-ExERH5o3oo6jFOdkvMP3gytTCQ9Ksi5PtylclJWghr7k7m3o2U5QrwtdiJkOxLOH4ghr0EKhpqGefzGz1VvVJg==", - "dev": true - }, - "debug": { - "version": "4.3.3", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.3.tgz", - "integrity": "sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==", + "@salesforce/plugin-generator": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-generator/-/plugin-generator-1.2.2.tgz", + "integrity": "sha512-ua6Brn/vzO4SqspF44dGOcmHP6ezgblL2BqV9Vul/Ze8ZOeaJvfgBA0BUrZSB3TVd9BrugwxoXa12T6F2qJRaw==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.0", + "@oclif/config": "^1.17.0", + "@oclif/errors": "^1.3.4", + "@oclif/fixpack": "^2.3.0", + "@oclif/plugin-help": "^3.2.2", + "@oclif/plugin-not-found": "^1.2.2", + "@oclif/plugin-warn-if-update-available": "^1.5.4", + "concurrently": "^7.0.0", + "debug": "^4.3.1", + "eslint-config-xo": "^0.36.0", + "eslint-config-xo-space": "^0.27.0", + "lodash": "^4.17.21", + "mkdirp": "^1.0.4", + "rimraf": "^3.0.2", + "sort-pjson": "^1.0.3", + "tslib": "^2.2.0", + "yeoman-environment": "^2.4.0", + "yeoman-generator": "^4.0.1", + "yosay": "^2.0.2" + }, + "dependencies": { + "@oclif/plugin-help": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-3.3.1.tgz", + "integrity": "sha512-QuSiseNRJygaqAdABYFWn/H1CwIZCp9zp/PLid6yXvy6VcQV7OenEFF5XuYaCvSARe2Tg9r8Jqls5+fw1A9CbQ==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/config": "1.18.2", + "@oclif/errors": "1.3.5", + "@oclif/help": "^1.0.1", + "chalk": "^4.1.2", + "indent-string": "^4.0.0", + "lodash": "^4.17.21", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "widest-line": "^3.1.0", + "wrap-ansi": "^6.2.0" + }, + "dependencies": { + "@oclif/config": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.2.tgz", + "integrity": "sha512-cE3qfHWv8hGRCP31j7fIS7BfCflm/BNZ2HNqHexH+fDrdF2f1D5S8VmXWLC77ffv3oDvWyvE9AZeR0RfmHCCaA==", + "dev": true, + "requires": { + "@oclif/errors": "^1.3.3", + "@oclif/parser": "^3.8.0", + "debug": "^4.1.1", + "globby": "^11.0.1", + "is-wsl": "^2.1.1", + "tslib": "^2.0.0" + } + } + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + } + } + } + }, + "@salesforce/plugin-info": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-info/-/plugin-info-1.3.1.tgz", + "integrity": "sha512-fL0Kq4o65IjXAs+O8yFYyc0fHQycOUEcRIm96w9ckcAv6hcDu3lnIv+63KuMSR7q/S6sG/W/TE7O/wOLCMwzCw==", "dev": true, "requires": { - "ms": "2.1.2" + "@oclif/config": "^1", + "@oclif/plugin-help": "^3.3.1", + "@salesforce/command": "^4.1.5", + "@salesforce/core": "^2.29.0", + "@salesforce/kit": "^1.5.17", + "got": "^11.8.2", + "marked": "^4.0.1", + "marked-terminal": "^4.2.0", + "proxy-agent": "^5.0.0", + "proxy-from-env": "^1.1.0", + "semver": "^7.3.5", + "tslib": "^2" + }, + "dependencies": { + "@oclif/plugin-help": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-3.3.1.tgz", + "integrity": "sha512-QuSiseNRJygaqAdABYFWn/H1CwIZCp9zp/PLid6yXvy6VcQV7OenEFF5XuYaCvSARe2Tg9r8Jqls5+fw1A9CbQ==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/config": "1.18.2", + "@oclif/errors": "1.3.5", + "@oclif/help": "^1.0.1", + "chalk": "^4.1.2", + "indent-string": "^4.0.0", + "lodash": "^4.17.21", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "widest-line": "^3.1.0", + "wrap-ansi": "^6.2.0" + }, + "dependencies": { + "@oclif/config": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.2.tgz", + "integrity": "sha512-cE3qfHWv8hGRCP31j7fIS7BfCflm/BNZ2HNqHexH+fDrdF2f1D5S8VmXWLC77ffv3oDvWyvE9AZeR0RfmHCCaA==", + "dev": true, + "requires": { + "@oclif/errors": "^1.3.3", + "@oclif/parser": "^3.8.0", + "debug": "^4.1.1", + "globby": "^11.0.1", + "is-wsl": "^2.1.1", + "tslib": "^2.0.0" + } + } + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "got": { + "version": "11.8.3", + "resolved": "https://registry.npmjs.org/got/-/got-11.8.3.tgz", + "integrity": "sha512-7gtQ5KiPh1RtGS9/Jbv1ofDpBFuq42gyfEib+ejaRBJuj/3tQFeR5+gw57e4ipaU8c/rCjvX6fkQz2lyDlGAOg==", + "dev": true, + "requires": { + "@sindresorhus/is": "^4.0.0", + "@szmarczak/http-timer": "^4.0.5", + "@types/cacheable-request": "^6.0.1", + "@types/responselike": "^1.0.0", + "cacheable-lookup": "^5.0.3", + "cacheable-request": "^7.0.2", + "decompress-response": "^6.0.0", + "http2-wrapper": "^1.0.0-beta.5.2", + "lowercase-keys": "^2.0.0", + "p-cancelable": "^2.0.0", + "responselike": "^2.0.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "lowercase-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", + "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + } + } } }, - "debuglog": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/debuglog/-/debuglog-1.0.1.tgz", - "integrity": "sha1-qiT/uaw9+aI1GDfPstJ5NgzXhJI=", - "dev": true - }, - "decimal.js": { - "version": "10.3.1", - "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.3.1.tgz", - "integrity": "sha512-V0pfhfr8suzyPGOx3nmq4aHqabehUZn6Ch9kyFpV79TGDTWFmHqUqXdabR7QHqxzrYolF4+tVmJhUG4OURg5dQ==", - "dev": true + "@salesforce/plugin-limits": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-limits/-/plugin-limits-1.3.0.tgz", + "integrity": "sha512-W3XN42SZzwSL+xattZdGA23Oy7Bg523n1FKEZg4vL6V8kSrP/C8AIm3U4UL5L9vBM89pCjCoDNgQ2ANy3r4BDg==", + "dev": true, + "requires": { + "@oclif/config": "^1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.31.0", + "tslib": "^2" + } }, - "decompress-response": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", - "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", + "@salesforce/plugin-org": { + "version": "1.11.2", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-org/-/plugin-org-1.11.2.tgz", + "integrity": "sha512-vHex65zx7Jd7FNCcBcFo12QymVHxJUF1XZ4O+MLMP6DEHnhiiQYo75bXcBBuumj+SAIfJi/RifYLpPRbyU/1Lw==", + "dev": true, "requires": { - "mimic-response": "^3.1.0" + "@oclif/config": "^1.18.1", + "@salesforce/command": "^4.2.2", + "@salesforce/core": "^2.35.0", + "@salesforce/kit": "^1.5.17", + "open": "8.4.0", + "tslib": "^2" }, "dependencies": { - "mimic-response": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", - "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==" + "open": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/open/-/open-8.4.0.tgz", + "integrity": "sha512-XgFPPM+B28FtCCgSb9I+s9szOC1vZRSwgWsRUA5ylIxRTgKozqjOCrVOqGsYABPYK5qnfqClxZTFBa8PKt2v6Q==", + "dev": true, + "requires": { + "define-lazy-prop": "^2.0.0", + "is-docker": "^2.1.1", + "is-wsl": "^2.2.0" + } } } }, - "dedent": { - "version": "0.7.0", - "resolved": "https://registry.npmjs.org/dedent/-/dedent-0.7.0.tgz", - "integrity": "sha1-JJXduvbrh0q7Dhvp3yLS5aVEMmw=", - "dev": true - }, - "deep-extend": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", - "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", - "dev": true - }, - "deep-is": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", - "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", - "dev": true - }, - "deepmerge": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", - "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==", - "dev": true - }, - "define-properties": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", - "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", + "@salesforce/plugin-schema": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-schema/-/plugin-schema-1.1.0.tgz", + "integrity": "sha512-fAYVrPnlZvWalqS3PrtrxID2KeXi2W9M7OhysctVWJUpBkoi3r1vCAK8RXCCP31H/KHt3PiuOyoyn4+8vfCgDA==", "dev": true, "requires": { - "object-keys": "^1.0.12" + "@oclif/config": "^1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.31.0", + "tslib": "^2" } }, - "delayed-stream": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", - "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", - "dev": true - }, - "detect-newline": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", - "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", - "dev": true - }, - "dezalgo": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.3.tgz", - "integrity": "sha1-f3Qt4Gb8dIvI24IFad3c5Jvw1FY=", + "@salesforce/plugin-source": { + "version": "1.9.2", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-source/-/plugin-source-1.9.2.tgz", + "integrity": "sha512-vNwMvlCtyoMWOu1q3zdsOQ4T58LEfLcrNdGWnhrDYKHuaEmQfezewNiE2R68hyDT9jJhDVK1OyM/0zpY832yXg==", "dev": true, "requires": { - "asap": "^2.0.0", - "wrappy": "1" - } - }, - "diff": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", - "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", - "dev": true - }, - "diff-sequences": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-27.4.0.tgz", - "integrity": "sha512-YqiQzkrsmHMH5uuh8OdQFU9/ZpADnwzml8z0O5HvRNda+5UZsaX/xN+AAxfR2hWq1Y7HZnAzO9J5lJXOuDz2Ww==", - "dev": true - }, - "dir-glob": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", - "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", - "dev": true, - "requires": { - "path-type": "^4.0.0" - } - }, - "dmd": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/dmd/-/dmd-6.0.0.tgz", - "integrity": "sha512-PwWZlqZnJPETwqZZ70haRa+UDZcD5jeBD3ywW1Kf+jYYv0MHu/S7Ri9jsSoeTMwkcMVW9hXOMA1IZUMEufBhOg==", - "dev": true, - "requires": { - "array-back": "^5.0.0", - "cache-point": "^2.0.0", - "common-sequence": "^2.0.0", - "file-set": "^4.0.1", - "handlebars": "^4.7.7", - "marked": "^2.0.0", - "object-get": "^2.1.1", - "reduce-flatten": "^3.0.0", - "reduce-unique": "^2.0.1", - "reduce-without": "^1.0.1", - "test-value": "^3.0.0", - "walk-back": "^5.0.0" + "@oclif/config": "^1.18.3", + "@oclif/plugin-help": "^3.3.1", + "@salesforce/command": "^4.2.2", + "@salesforce/core": "^2.35.0", + "@salesforce/source-deploy-retrieve": "^5.12.5", + "@salesforce/source-tracking": "^1.3.1", + "chalk": "^4.1.2", + "cli-ux": "^5.6.3", + "got": "^11.8.3", + "open": "^8.4.0", + "proxy-agent": "^5.0.0", + "proxy-from-env": "^1.1.0", + "tslib": "^2" }, "dependencies": { - "array-back": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-5.0.0.tgz", - "integrity": "sha512-kgVWwJReZWmVuWOQKEOohXKJX+nD02JAZ54D1RRWlv8L0NebauKAaFxACKzB74RTclt1+WNz5KHaLRDAPZbDEw==", + "@oclif/plugin-help": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-3.3.1.tgz", + "integrity": "sha512-QuSiseNRJygaqAdABYFWn/H1CwIZCp9zp/PLid6yXvy6VcQV7OenEFF5XuYaCvSARe2Tg9r8Jqls5+fw1A9CbQ==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/config": "1.18.2", + "@oclif/errors": "1.3.5", + "@oclif/help": "^1.0.1", + "chalk": "^4.1.2", + "indent-string": "^4.0.0", + "lodash": "^4.17.21", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "widest-line": "^3.1.0", + "wrap-ansi": "^6.2.0" + }, + "dependencies": { + "@oclif/config": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.2.tgz", + "integrity": "sha512-cE3qfHWv8hGRCP31j7fIS7BfCflm/BNZ2HNqHexH+fDrdF2f1D5S8VmXWLC77ffv3oDvWyvE9AZeR0RfmHCCaA==", + "dev": true, + "requires": { + "@oclif/errors": "^1.3.3", + "@oclif/parser": "^3.8.0", + "debug": "^4.1.1", + "globby": "^11.0.1", + "is-wsl": "^2.1.1", + "tslib": "^2.0.0" + } + } + } + }, + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "cli-ux": { + "version": "5.6.7", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.7.tgz", + "integrity": "sha512-dsKAurMNyFDnO6X1TiiRNiVbL90XReLKcvIq4H777NMqXGBxBws23ag8ubCJE97vVZEgWG2eSUhsyLf63Jv8+g==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true }, - "marked": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/marked/-/marked-2.1.3.tgz", - "integrity": "sha512-/Q+7MGzaETqifOMWYEA7HVMaZb4XbcRfaOzcSsHZEith83KGlvaSG33u0SKu89Mj5h+T8V2hM+8O45Qc5XTgwA==", + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", "dev": true }, - "reduce-flatten": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/reduce-flatten/-/reduce-flatten-3.0.1.tgz", - "integrity": "sha512-bYo+97BmUUOzg09XwfkwALt4PQH1M5L0wzKerBt6WLm3Fhdd43mMS89HiT1B9pJIqko/6lWx3OnV4J9f2Kqp5Q==", + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "got": { + "version": "11.8.3", + "resolved": "https://registry.npmjs.org/got/-/got-11.8.3.tgz", + "integrity": "sha512-7gtQ5KiPh1RtGS9/Jbv1ofDpBFuq42gyfEib+ejaRBJuj/3tQFeR5+gw57e4ipaU8c/rCjvX6fkQz2lyDlGAOg==", + "dev": true, + "requires": { + "@sindresorhus/is": "^4.0.0", + "@szmarczak/http-timer": "^4.0.5", + "@types/cacheable-request": "^6.0.1", + "@types/responselike": "^1.0.0", + "cacheable-lookup": "^5.0.3", + "cacheable-request": "^7.0.2", + "decompress-response": "^6.0.0", + "http2-wrapper": "^1.0.0-beta.5.2", + "lowercase-keys": "^2.0.0", + "p-cancelable": "^2.0.0", + "responselike": "^2.0.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "lowercase-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", + "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", "dev": true + }, + "open": { + "version": "8.4.0", + "resolved": "https://registry.npmjs.org/open/-/open-8.4.0.tgz", + "integrity": "sha512-XgFPPM+B28FtCCgSb9I+s9szOC1vZRSwgWsRUA5ylIxRTgKozqjOCrVOqGsYABPYK5qnfqClxZTFBa8PKt2v6Q==", + "dev": true, + "requires": { + "define-lazy-prop": "^2.0.0", + "is-docker": "^2.1.1", + "is-wsl": "^2.2.0" + } + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + } } } }, - "doctrine": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", - "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=", + "@salesforce/plugin-telemetry": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-telemetry/-/plugin-telemetry-1.4.0.tgz", + "integrity": "sha512-vroDey9dTomc6mMWWufzN2mKXsWzLR067z2263TPEFr3Hgt0qQjE0K4eXa5rjbs0OOpm6dGGwisa9sT6wkoseA==", "dev": true, "requires": { - "esutils": "^2.0.2", - "isarray": "^1.0.0" + "@oclif/config": "^1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.31.0", + "@salesforce/telemetry": "^3.1.0", + "tslib": "^2" } }, - "domexception": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/domexception/-/domexception-2.0.1.tgz", - "integrity": "sha512-yxJ2mFy/sibVQlu5qHjOkf9J3K6zgmCxgJ94u2EdvDOV09H+32LtRswEcUsmUWN72pVLOEnTSRaIVVzVQgS0dg==", + "@salesforce/plugin-templates": { + "version": "54.3.0", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-templates/-/plugin-templates-54.3.0.tgz", + "integrity": "sha512-gSsFsaO7peDcI+2gp1i2QvJ6RGbHCZKBIEHlpviwtMGJzPAV0eA3alvvBLTg7A/feZK+Kv41pZM2x5Qo1DUg4A==", "dev": true, "requires": { - "webidl-conversions": "^5.0.0" + "@oclif/command": "^1", + "@oclif/config": "^1", + "@oclif/errors": "^1", + "@salesforce/command": "^4.2.1", + "@salesforce/core": "^2.33.1", + "@salesforce/templates": "54.3.0", + "tslib": "^1", + "yeoman-environment": "2.4.0", + "yeoman-generator": "^4.8.2" }, "dependencies": { - "webidl-conversions": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-5.0.0.tgz", - "integrity": "sha512-VlZwKPCkYKxQgeSbH5EyngOmRp7Ww7I9rQLERETtf5ofd9pGeswWiOtogpEO850jziPRarreGxn5QIiTqpb2wA==", - "dev": true - } - } - }, - "dtrace-provider": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/dtrace-provider/-/dtrace-provider-0.6.0.tgz", - "integrity": "sha1-CweNVReTfYcxAUUtkUZzdVe3XlE=", - "dev": true, - "optional": true, - "requires": { - "nan": "^2.0.8" - } - }, - "eastasianwidth": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", - "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", - "dev": true - }, - "ecc-jsbn": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", - "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", - "dev": true, - "requires": { - "jsbn": "~0.1.0", - "safer-buffer": "^2.1.0" - } - }, - "ecdsa-sig-formatter": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", - "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", - "dev": true, - "requires": { - "safe-buffer": "^5.0.1" - } - }, - "electron-to-chromium": { - "version": "1.4.31", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.31.tgz", - "integrity": "sha512-t3XVQtk+Frkv6aTD4RRk0OqosU+VLe1dQFW83MDer78ZD6a52frgXuYOIsLYTQiH2Lm+JB2OKYcn7zrX+YGAiQ==", - "dev": true - }, - "emittery": { - "version": "0.8.1", - "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.8.1.tgz", - "integrity": "sha512-uDfvUjVrfGJJhymx/kz6prltenw1u7WrCg1oa94zYY8xxVpLLUu045LAT0dhDZdXG58/EpPL/5kA180fQ/qudg==", - "dev": true - }, - "emoji-regex": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true - }, - "end-of-stream": { - "version": "1.4.4", - "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", - "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", - "dev": true, - "requires": { - "once": "^1.4.0" - } - }, - "entities": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/entities/-/entities-2.0.3.tgz", - "integrity": "sha512-MyoZ0jgnLvB2X3Lg5HqpFmn1kybDiIfEQmKzTb5apr51Rb+T3KdmMiqa70T+bhGnyv7bQ6WMj2QMHpGMmlrUYQ==", - "dev": true - }, - "error-ex": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", - "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", - "dev": true, - "requires": { - "is-arrayish": "^0.2.1" - } - }, - "es-abstract": { - "version": "1.19.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.19.1.tgz", - "integrity": "sha512-2vJ6tjA/UfqLm2MPs7jxVybLoB8i1t1Jd9R3kISld20sIxPcTbLuggQOUxeWeAvIUkduv/CfMjuh4WmiXr2v9w==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "get-intrinsic": "^1.1.1", - "get-symbol-description": "^1.0.0", - "has": "^1.0.3", - "has-symbols": "^1.0.2", - "internal-slot": "^1.0.3", - "is-callable": "^1.2.4", - "is-negative-zero": "^2.0.1", - "is-regex": "^1.1.4", - "is-shared-array-buffer": "^1.0.1", - "is-string": "^1.0.7", - "is-weakref": "^1.0.1", - "object-inspect": "^1.11.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.2", - "string.prototype.trimend": "^1.0.4", - "string.prototype.trimstart": "^1.0.4", - "unbox-primitive": "^1.0.1" - } - }, - "es-to-primitive": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", - "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", - "dev": true, - "requires": { - "is-callable": "^1.1.4", - "is-date-object": "^1.0.1", - "is-symbol": "^1.0.2" - } - }, - "escalade": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", - "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", - "dev": true - }, - "escape-string-regexp": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", - "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", - "dev": true - }, - "escodegen": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-2.0.0.tgz", - "integrity": "sha512-mmHKys/C8BFUGI+MAWNcSYoORYLMdPzjrknd2Vc+bUsjN5bXcr8EhrNB+UTqfL1y3I9c4fw2ihgtMPQLBRiQxw==", - "dev": true, - "requires": { - "esprima": "^4.0.1", - "estraverse": "^5.2.0", - "esutils": "^2.0.2", - "optionator": "^0.8.1", - "source-map": "~0.6.1" - }, - "dependencies": { - "estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", "dev": true }, - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true, - "optional": true - } - } - }, - "eslint": { - "version": "8.8.0", - "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.8.0.tgz", - "integrity": "sha512-H3KXAzQGBH1plhYS3okDix2ZthuYJlQQEGE5k0IKuEqUSiyu4AmxxlJ2MtTYeJ3xB4jDhcYCwGOg2TXYdnDXlQ==", - "dev": true, - "requires": { - "@eslint/eslintrc": "^1.0.5", - "@humanwhocodes/config-array": "^0.9.2", - "ajv": "^6.10.0", - "chalk": "^4.0.0", - "cross-spawn": "^7.0.2", - "debug": "^4.3.2", - "doctrine": "^3.0.0", - "escape-string-regexp": "^4.0.0", - "eslint-scope": "^7.1.0", - "eslint-utils": "^3.0.0", - "eslint-visitor-keys": "^3.2.0", - "espree": "^9.3.0", - "esquery": "^1.4.0", - "esutils": "^2.0.2", - "fast-deep-equal": "^3.1.3", - "file-entry-cache": "^6.0.1", - "functional-red-black-tree": "^1.0.1", - "glob-parent": "^6.0.1", - "globals": "^13.6.0", - "ignore": "^5.2.0", - "import-fresh": "^3.0.0", - "imurmurhash": "^0.1.4", - "is-glob": "^4.0.0", - "js-yaml": "^4.1.0", - "json-stable-stringify-without-jsonify": "^1.0.1", - "levn": "^0.4.1", - "lodash.merge": "^4.6.2", - "minimatch": "^3.0.4", - "natural-compare": "^1.4.0", - "optionator": "^0.9.1", - "regexpp": "^3.2.0", - "strip-ansi": "^6.0.1", - "strip-json-comments": "^3.1.0", - "text-table": "^0.2.0", - "v8-compile-cache": "^2.0.3" - }, - "dependencies": { - "acorn": { - "version": "8.7.0", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.7.0.tgz", - "integrity": "sha512-V/LGr1APy+PXIwKebEWrkZPwoeoF+w1jiOBUmuxuiUIaOHtob8Qc9BTrYo7VuI5fR8tqsy+buA2WFooR5olqvQ==", + "ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", "dev": true }, - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", "dev": true, "requires": { - "color-convert": "^2.0.1" + "array-uniq": "^1.0.1" } }, - "argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true - }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", "dev": true, "requires": { - "color-name": "~1.1.4" + "restore-cursor": "^2.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "cli-width": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", + "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", "dev": true }, - "doctrine": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", - "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", "dev": true, "requires": { - "esutils": "^2.0.2" + "ms": "^2.1.1" } }, - "escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", "dev": true }, - "eslint-scope": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.1.0.tgz", - "integrity": "sha512-aWwkhnS0qAXqNOgKOK0dJ2nvzEbhEvpy8OlJ9kZ0FeZnA6zpjv1/Vei+puGFFX7zkPCkHHXb7IDX3A+7yPrRWg==", + "dir-glob": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.0.0.tgz", + "integrity": "sha512-37qirFDz8cA5fimp9feo43fSuRo2gHwaIn6dXL8Ber1dGwUosDrGZeCCXq57WnIqE4aQ+u3eQZzsk1yOzhdwag==", "dev": true, "requires": { - "esrecurse": "^4.3.0", - "estraverse": "^5.2.0" + "arrify": "^1.0.1", + "path-type": "^3.0.0" } }, - "eslint-utils": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz", - "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==", + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", "dev": true, "requires": { - "eslint-visitor-keys": "^2.0.0" - }, - "dependencies": { - "eslint-visitor-keys": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", - "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", - "dev": true - } + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" } }, - "eslint-visitor-keys": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.2.0.tgz", - "integrity": "sha512-IOzT0X126zn7ALX0dwFiUQEdsfzrm4+ISsQS8nukaJXwEyYKRSnEIIDULYg1mCtGp7UUXgfGl7BIolXREQK+XQ==", - "dev": true - }, - "espree": { - "version": "9.3.0", - "resolved": "https://registry.npmjs.org/espree/-/espree-9.3.0.tgz", - "integrity": "sha512-d/5nCsb0JcqsSEeQzFZ8DH1RmxPcglRWh24EFTlUEmCKoehXGdpsx0RkHDubqUI8LSAIKMQp4r9SzQ3n+sm4HQ==", + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", "dev": true, "requires": { - "acorn": "^8.7.0", - "acorn-jsx": "^5.3.1", - "eslint-visitor-keys": "^3.1.0" + "escape-string-regexp": "^1.0.5" } }, - "estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", - "dev": true - }, - "file-entry-cache": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", - "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", "dev": true, "requires": { - "flat-cache": "^3.0.4" + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } } }, - "flat-cache": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", - "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", "dev": true, "requires": { - "flatted": "^3.1.0", - "rimraf": "^3.0.2" + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } } }, - "flatted": { - "version": "3.2.5", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.5.tgz", - "integrity": "sha512-WIWGi2L3DyTUvUrwRKgGi9TwxQMUEqPOPQBVi71R96jZXJdFskXEmf54BoZaS1kknGODoIGASGEzBUYdyMCBJg==", - "dev": true - }, - "glob-parent": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", - "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "globby": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/globby/-/globby-8.0.2.tgz", + "integrity": "sha512-yTzMmKygLp8RUpG1Ymu2VXPSJQZjNAZPD4ywgYEaG7e4tBJeUQBO8OpXrf1RCNcEs5alsoJYPAMiIHP0cmeC7w==", "dev": true, "requires": { - "is-glob": "^4.0.3" + "array-union": "^1.0.1", + "dir-glob": "2.0.0", + "fast-glob": "^2.0.2", + "glob": "^7.1.2", + "ignore": "^3.3.5", + "pify": "^3.0.0", + "slash": "^1.0.0" } }, - "globals": { - "version": "13.12.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-13.12.0.tgz", - "integrity": "sha512-uS8X6lSKN2JumVoXrbUz+uG4BYG+eiawqm3qFcT7ammfbUHeCBoJMlHcec/S3krSk73/AE/f0szYFmgAA3kYZg==", + "grouped-queue": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/grouped-queue/-/grouped-queue-0.3.3.tgz", + "integrity": "sha1-wWfSpTGcWg4JZO9qJbfC34mWyFw=", "dev": true, "requires": { - "type-fest": "^0.20.2" + "lodash": "^4.17.2" } }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true - }, "ignore": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.0.tgz", - "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", "dev": true }, - "js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "dev": true, - "requires": { - "argparse": "^2.0.1" - } - }, - "levn": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", - "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "inquirer": { + "version": "6.5.2", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-6.5.2.tgz", + "integrity": "sha512-cntlB5ghuB0iuO65Ovoi8ogLHiWGs/5yNrtUcKjFhSSiVeAIVpD7koaSU9RM8mpXw5YDi9RdYXGQMaOURB7ycQ==", "dev": true, "requires": { - "prelude-ls": "^1.2.1", - "type-check": "~0.4.0" + "ansi-escapes": "^3.2.0", + "chalk": "^2.4.2", + "cli-cursor": "^2.1.0", + "cli-width": "^2.0.0", + "external-editor": "^3.0.3", + "figures": "^2.0.0", + "lodash": "^4.17.12", + "mute-stream": "0.0.7", + "run-async": "^2.2.0", + "rxjs": "^6.4.0", + "string-width": "^2.1.0", + "strip-ansi": "^5.1.0", + "through": "^2.3.6" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } } }, - "optionator": { - "version": "0.9.1", - "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz", - "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==", + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", "dev": true, "requires": { - "deep-is": "^0.1.3", - "fast-levenshtein": "^2.0.6", - "levn": "^0.4.1", - "prelude-ls": "^1.2.1", - "type-check": "^0.4.0", - "word-wrap": "^1.2.3" + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } } }, - "prelude-ls": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", - "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", "dev": true, "requires": { - "has-flag": "^4.0.0" + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" } }, - "type-check": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", - "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", - "dev": true, - "requires": { - "prelude-ls": "^1.2.1" - } + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true }, - "type-fest": { - "version": "0.20.2", - "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", - "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", "dev": true - } - } - }, - "eslint-config-prettier": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.3.0.tgz", - "integrity": "sha512-BgZuLUSeKzvlL/VUjx/Yb787VQ26RU3gGjA3iiFvdsp/2bMfVIWUVP7tjxtjS0e+HP409cPlPvNkQloz8C91ew==", - "dev": true - }, - "eslint-import-resolver-node": { - "version": "0.3.6", - "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.6.tgz", - "integrity": "sha512-0En0w03NRVMn9Uiyn8YRPDKvWjxCWkslUEhGNTdGx15RvPJYQ+lbOlqrlNI2vEAs4pDYK4f/HN2TbDmk5TP0iw==", - "dev": true, - "requires": { - "debug": "^3.2.7", - "resolve": "^1.20.0" - }, - "dependencies": { - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", "dev": true, "requires": { - "ms": "^2.1.1" + "mimic-fn": "^1.0.0" } - } - } - }, - "eslint-module-utils": { - "version": "2.7.2", - "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.7.2.tgz", - "integrity": "sha512-zquepFnWCY2ISMFwD/DqzaM++H+7PDzOpUvotJWm/y1BAFt5R4oeULgdrTejKqLkz7MA/tgstsUMNYc7wNdTrg==", - "dev": true, - "requires": { - "debug": "^3.2.7", - "find-up": "^2.1.0" - }, - "dependencies": { - "debug": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", - "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", "dev": true, "requires": { - "ms": "^2.1.1" + "pify": "^3.0.0" } }, - "find-up": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", - "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", "dev": true, "requires": { - "locate-path": "^2.0.0" + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" } }, - "locate-path": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", - "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "slash": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", + "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=", + "dev": true + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", "dev": true, "requires": { - "p-locate": "^2.0.0", - "path-exists": "^3.0.0" + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" } }, - "p-limit": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", - "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, "requires": { - "p-try": "^1.0.0" + "ansi-regex": "^3.0.0" } }, - "p-locate": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", - "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", "dev": true, "requires": { - "p-limit": "^1.1.0" + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" } }, - "p-try": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", - "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", "dev": true }, - "path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "untildify": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/untildify/-/untildify-3.0.3.tgz", + "integrity": "sha512-iSk/J8efr8uPT/Z4eSUywnqyrQU7DSdMfdqK4iWEaUVVmcP5JcnpRqmVMwcwcnmI1ATFNgC5V90u09tBynNFKA==", "dev": true - } - } - }, - "eslint-plugin-compat": { - "version": "3.13.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-compat/-/eslint-plugin-compat-3.13.0.tgz", - "integrity": "sha512-cv8IYMuTXm7PIjMVDN2y4k/KVnKZmoNGHNq27/9dLstOLydKblieIv+oe2BN2WthuXnFNhaNvv3N1Bvl4dbIGA==", - "dev": true, - "requires": { - "@mdn/browser-compat-data": "^3.3.14", - "ast-metadata-inferer": "^0.7.0", - "browserslist": "^4.16.8", - "caniuse-lite": "^1.0.30001251", - "core-js": "^3.16.2", - "find-up": "^5.0.0", - "lodash.memoize": "4.1.2", - "semver": "7.3.5" - }, - "dependencies": { - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + }, + "yeoman-environment": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/yeoman-environment/-/yeoman-environment-2.4.0.tgz", + "integrity": "sha512-SsvoL0RNAFIX69eFxkUhwKUN2hG1UwUjxrcP+T2ytwdhqC/kHdnFOH2SXdtSN1Ju4aO4xuimmzfRoheYY88RuA==", "dev": true, "requires": { - "lru-cache": "^6.0.0" + "chalk": "^2.4.1", + "cross-spawn": "^6.0.5", + "debug": "^3.1.0", + "diff": "^3.5.0", + "escape-string-regexp": "^1.0.2", + "globby": "^8.0.1", + "grouped-queue": "^0.3.3", + "inquirer": "^6.0.0", + "is-scoped": "^1.0.0", + "lodash": "^4.17.10", + "log-symbols": "^2.2.0", + "mem-fs": "^1.1.0", + "strip-ansi": "^4.0.0", + "text-table": "^0.2.0", + "untildify": "^3.0.3" } } } }, - "eslint-plugin-import": { - "version": "2.25.4", - "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.25.4.tgz", - "integrity": "sha512-/KJBASVFxpu0xg1kIBn9AUa8hQVnszpwgE7Ld0lKAlx7Ie87yzEzCgSkekt+le/YVhiaosO4Y14GDAOc41nfxA==", + "@salesforce/plugin-trust": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-trust/-/plugin-trust-1.1.0.tgz", + "integrity": "sha512-hk9A2GwNHgzho5E3ZGw+hMUlqRTaRXHNLpiIQwZjLOnOPYS3lT6dGR9vidQb6WXs7zYZIW2feI+sN36FZjcIDA==", "dev": true, "requires": { - "array-includes": "^3.1.4", - "array.prototype.flat": "^1.2.5", - "debug": "^2.6.9", - "doctrine": "^2.1.0", - "eslint-import-resolver-node": "^0.3.6", - "eslint-module-utils": "^2.7.2", - "has": "^1.0.3", - "is-core-module": "^2.8.0", - "is-glob": "^4.0.3", - "minimatch": "^3.0.4", - "object.values": "^1.1.5", - "resolve": "^1.20.0", - "tsconfig-paths": "^3.12.0" - }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "doctrine": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", - "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", - "dev": true, - "requires": { - "esutils": "^2.0.2" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } + "@oclif/config": "^1.18.1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.31.0", + "npm": "^7.21.0", + "npm-run-path": "^4.0.1", + "shelljs": "^0.8.4", + "tslib": "^2" } }, - "eslint-plugin-jest": { - "version": "23.8.2", - "resolved": "https://registry.npmjs.org/eslint-plugin-jest/-/eslint-plugin-jest-23.8.2.tgz", - "integrity": "sha512-xwbnvOsotSV27MtAe7s8uGWOori0nUsrXh2f1EnpmXua8sDfY6VZhHAhHg2sqK7HBNycRQExF074XSZ7DvfoFg==", + "@salesforce/plugin-user": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/@salesforce/plugin-user/-/plugin-user-1.7.1.tgz", + "integrity": "sha512-nM0j/AqtJYV9VH67DSQ/TkXlB3J1SxlngOH8byzyPWTAhbrNDFxo44Up7nsiLYyrdMrKrmJ3eOWjDm9rKRFUtA==", "dev": true, "requires": { - "@typescript-eslint/experimental-utils": "^2.5.0" + "@oclif/config": "^1.18.1", + "@salesforce/command": "^4.2.0", + "@salesforce/core": "^2.31.0", + "@salesforce/kit": "^1.5.18", + "tslib": "^2" } }, - "eslint-restricted-globals": { - "version": "0.2.0", - "resolved": "https://registry.npmjs.org/eslint-restricted-globals/-/eslint-restricted-globals-0.2.0.tgz", - "integrity": "sha512-kwYJALm5KS2QW3Mc1PgObO4V+pTR6RQtRT65L1GQILlEnAhabUQqGAX7/qUjoQR4KZJKehWpBtyDEiDecwmY9A==", - "dev": true - }, - "eslint-scope": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", - "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "@salesforce/require-analytics": { + "version": "0.9.21", + "resolved": "https://registry.npmjs.org/@salesforce/require-analytics/-/require-analytics-0.9.21.tgz", + "integrity": "sha512-BACe6i2c4AwBD72jsOWN3pbpVl0YMZlJjgLFWskigvKCSypW8oXK+klabF1lBPS0rIssAvyIBs9HGKVrXtsqhg==", "dev": true, "requires": { - "esrecurse": "^4.3.0", - "estraverse": "^4.1.1" + "tslib": "^2.2.0" } }, - "eslint-utils": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", - "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", + "@salesforce/schemas": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@salesforce/schemas/-/schemas-1.1.0.tgz", + "integrity": "sha512-6D7DvE6nFxpLyyTnrOIbbAeCJw2r/EpinFAcMh6gU0gA/CGfSbwV/8uR3uHLYL2zCyCZLH8jJ4dZ3BzCMqc+Eg==", + "dev": true + }, + "@salesforce/sfdx-lwc-jest": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@salesforce/sfdx-lwc-jest/-/sfdx-lwc-jest-1.1.0.tgz", + "integrity": "sha512-bYhvs9Th4qMjk4Vs66GDwloVh2LAcV8cg/j0WWbIug8MiI+KOSZL0IWZA7fqr8vXkSdArt5WjoTv4ckzfklzQA==", "dev": true, "requires": { - "eslint-visitor-keys": "^1.1.0" + "@lwc/compiler": "2.2.11", + "@lwc/engine-dom": "2.2.11", + "@lwc/jest-preset": "11.2.0", + "@lwc/jest-resolver": "11.2.0", + "@lwc/jest-serializer": "11.2.0", + "@lwc/jest-transformer": "11.2.0", + "@lwc/module-resolver": "2.2.11", + "@lwc/synthetic-shadow": "2.2.11", + "@lwc/wire-service": "2.2.11", + "@salesforce/wire-service-jest-util": "4.0.0", + "chalk": "~4.1.1", + "fast-glob": "^3.2.7", + "jest": "27.0.6", + "yargs": "~17.0.1" }, "dependencies": { - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", - "dev": true - } - } - }, - "eslint-visitor-keys": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", - "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", - "dev": true - }, - "espree": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/espree/-/espree-6.2.1.tgz", - "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", - "dev": true, - "requires": { - "acorn": "^7.1.1", - "acorn-jsx": "^5.2.0", - "eslint-visitor-keys": "^1.1.0" - }, - "dependencies": { - "acorn": { - "version": "7.4.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", - "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", - "dev": true + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } }, - "eslint-visitor-keys": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", - "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", "dev": true - } - } - }, - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true - }, - "esquery": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.4.0.tgz", - "integrity": "sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w==", - "dev": true, - "requires": { - "estraverse": "^5.1.0" - }, - "dependencies": { - "estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "yargs": { + "version": "17.0.1", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.0.1.tgz", + "integrity": "sha512-xBBulfCc8Y6gLFcrPvtqKz9hz8SO0l1Ni8GgDekvBX2ro0HRQImDGnikfc33cgzcYUSncapnNcZDjVFIH3f6KQ==", + "dev": true, + "requires": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + } } } }, - "esrecurse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", - "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "@salesforce/sfdx-plugin-lwc-test": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/@salesforce/sfdx-plugin-lwc-test/-/sfdx-plugin-lwc-test-0.1.7.tgz", + "integrity": "sha512-Cipsf/VimgyoOWWSlnaUKh58oCCZQ7wyGmvTfGY5QJkxUGVgUvXzIlTWEHi+oLzSMH7GZQDHU2KLWWq82rgd4A==", "dev": true, "requires": { - "estraverse": "^5.2.0" + "@oclif/command": "^1.5.19", + "@oclif/config": "1.13.3", + "@oclif/errors": "^1.2.2", + "@salesforce/command": "^3.0.0", + "@salesforce/core": "^2.2.0", + "@salesforce/kit": "^1.2.1", + "semver-compare": "^1.0.0", + "signal-exit": "^3.0.2", + "tslib": "^1.10.0" }, "dependencies": { - "estraverse": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", - "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "@oclif/config": { + "version": "1.13.3", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.13.3.tgz", + "integrity": "sha512-qs5XvGRw+1M41abOKCjd0uoeHCgsMxa2MurD2g2K8CtQlzlMXl0rW5idVeimIg5208LLuxkfzQo8TKAhhRCWLg==", + "dev": true, + "requires": { + "@oclif/parser": "^3.8.0", + "debug": "^4.1.1", + "tslib": "^1.9.3" + } + }, + "@salesforce/command": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-3.1.3.tgz", + "integrity": "sha512-Yg9lhl3ghwPN7WwqFmgfWIn6i7vz43WTpEsYsChz80bKORlVbDvhwPZQUj0XTv3DKDnPZVU8FFIDsrLSOa9G1A==", + "dev": true, + "requires": { + "@oclif/command": "^1.5.17", + "@oclif/errors": "^1.2.2", + "@oclif/parser": "^3.8.3", + "@oclif/plugin-help": "^2.2.0", + "@oclif/test": "^1.2.4", + "@salesforce/core": "^2.23.4", + "@salesforce/kit": "^1.2.2", + "@salesforce/ts-types": "^1.2.0", + "chalk": "^2.4.2", + "cli-ux": "^4.9.3" + } + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", "dev": true } } }, - "estraverse": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", - "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", - "dev": true - }, - "estree-walker": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", - "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", - "dev": true - }, - "esutils": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", - "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true - }, - "execa": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", - "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "@salesforce/sfdx-scanner": { + "version": "2.13.1", + "resolved": "https://registry.npmjs.org/@salesforce/sfdx-scanner/-/sfdx-scanner-2.13.1.tgz", + "integrity": "sha512-ln0Uo/+rNkBK3CBcBEKdTjcnc/t8uwefoi9S+7PyeBIrXQNyMNeBpWBZKyMekdSYj1E7y6uk9BwGGo69sknkGg==", "dev": true, "requires": { + "@lwc/eslint-plugin-lwc": "^0.10.0", + "@oclif/command": "^1", + "@oclif/config": "^1", + "@oclif/errors": "^1", + "@salesforce/command": "^3", + "@salesforce/core": "^2.1.6", + "@salesforce/eslint-config-lwc": "^0.7.0", + "@typescript-eslint/eslint-plugin": "^2.21.0", + "@typescript-eslint/parser": "^2.21.0", + "babel-eslint": "^10.1.0", "cross-spawn": "^7.0.3", - "get-stream": "^6.0.0", - "human-signals": "^2.1.0", - "is-stream": "^2.0.0", - "merge-stream": "^2.0.0", - "npm-run-path": "^4.0.1", - "onetime": "^5.1.2", - "signal-exit": "^3.0.3", - "strip-final-newline": "^2.0.0" - } - }, - "exit": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", - "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=", - "dev": true - }, - "exit-on-epipe": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/exit-on-epipe/-/exit-on-epipe-1.0.1.tgz", - "integrity": "sha512-h2z5mrROTxce56S+pnvAV890uu7ls7f1kEvVGJbw1OlFH3/mlJ5bkXu0KRyW94v37zzHPiUd55iLn3DA7TjWpw==", - "dev": true - }, - "expect": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/expect/-/expect-27.4.2.tgz", - "integrity": "sha512-BjAXIDC6ZOW+WBFNg96J22D27Nq5ohn+oGcuP2rtOtcjuxNoV9McpQ60PcQWhdFOSBIQdR72e+4HdnbZTFSTyg==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "ansi-styles": "^5.0.0", - "jest-get-type": "^27.4.0", - "jest-matcher-utils": "^27.4.2", - "jest-message-util": "^27.4.2", - "jest-regex-util": "^27.4.0" + "eslint": "^6.8.0", + "find-java-home": "1.1.0", + "globby": "^11.0.0", + "html-escaper": "^3.0.0", + "is-zip": "^1.0.0", + "isbinaryfile": "^4.0.8", + "mustache": "^4.0.1", + "node-stream-zip": "1.13.2", + "normalize-path": "^3.0.0", + "picomatch": "^2.2.2", + "reflect-metadata": "^0.1.13", + "retire": "^2.2.5", + "semver": "^7.3.4", + "ts-node": "^8", + "tslib": "^2", + "tsyringe": "^4.1.0", + "typescript": "^3.8.2", + "untildify": "^4.0.0", + "word-wrap": "^1.2.3", + "xml-js": "^1.6.11" }, "dependencies": { - "ansi-styles": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-5.2.0.tgz", - "integrity": "sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA==", - "dev": true - } - } - }, - "extend": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", - "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", - "dev": true - }, - "external-editor": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", - "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", - "dev": true, - "requires": { - "chardet": "^0.7.0", - "iconv-lite": "^0.4.24", - "tmp": "^0.0.33" - } - }, - "extract-stack": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-1.0.0.tgz", - "integrity": "sha1-uXrK+UQe6iMyUpYktzL8WhyBZfo=", - "dev": true - }, - "extsprintf": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", - "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", + "@lwc/eslint-plugin-lwc": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/@lwc/eslint-plugin-lwc/-/eslint-plugin-lwc-0.10.0.tgz", + "integrity": "sha512-UsosUow0xWrzzcSK1fWDWHKBT81/pJz1/icfv7w8T+BCE9G4Kb4TwJexHLOIZvY7e/dv4Fjme4lTOWehuNE5Sg==", + "dev": true, + "requires": { + "minimatch": "^3.0.4" + } + }, + "@salesforce/command": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/@salesforce/command/-/command-3.1.3.tgz", + "integrity": "sha512-Yg9lhl3ghwPN7WwqFmgfWIn6i7vz43WTpEsYsChz80bKORlVbDvhwPZQUj0XTv3DKDnPZVU8FFIDsrLSOa9G1A==", + "dev": true, + "requires": { + "@oclif/command": "^1.5.17", + "@oclif/errors": "^1.2.2", + "@oclif/parser": "^3.8.3", + "@oclif/plugin-help": "^2.2.0", + "@oclif/test": "^1.2.4", + "@salesforce/core": "^2.23.4", + "@salesforce/kit": "^1.2.2", + "@salesforce/ts-types": "^1.2.0", + "chalk": "^2.4.2", + "cli-ux": "^4.9.3" + } + }, + "@salesforce/eslint-config-lwc": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/@salesforce/eslint-config-lwc/-/eslint-config-lwc-0.7.0.tgz", + "integrity": "sha512-x87H9+amXcPDmOnOZoLdajp5yNZqv8ieyzfE4R8QDOJeZfj1X5I2AOSZYG0pUV51eUV/kOdGjAj28zU296t2yQ==", + "dev": true, + "requires": { + "@lwc/eslint-plugin-lwc": "~0.10.0", + "babel-eslint": "~10.1.0", + "eslint-plugin-import": "~2.20.2", + "eslint-plugin-jest": "~23.8.2", + "eslint-restricted-globals": "~0.2.0" + } + }, + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true + }, + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "eslint": { + "version": "6.8.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-6.8.0.tgz", + "integrity": "sha512-K+Iayyo2LtyYhDSYwz5D5QdWw0hCacNzyq1Y821Xna2xSJj7cijoLLYmLxTQgcgZ9mC61nryMy9S7GRbYpI5Ig==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "ajv": "^6.10.0", + "chalk": "^2.1.0", + "cross-spawn": "^6.0.5", + "debug": "^4.0.1", + "doctrine": "^3.0.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^1.4.3", + "eslint-visitor-keys": "^1.1.0", + "espree": "^6.1.2", + "esquery": "^1.0.1", + "esutils": "^2.0.2", + "file-entry-cache": "^5.0.1", + "functional-red-black-tree": "^1.0.1", + "glob-parent": "^5.0.0", + "globals": "^12.1.0", + "ignore": "^4.0.6", + "import-fresh": "^3.0.0", + "imurmurhash": "^0.1.4", + "inquirer": "^7.0.0", + "is-glob": "^4.0.0", + "js-yaml": "^3.13.1", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.3.0", + "lodash": "^4.17.14", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.1", + "natural-compare": "^1.4.0", + "optionator": "^0.8.3", + "progress": "^2.0.0", + "regexpp": "^2.0.1", + "semver": "^6.1.2", + "strip-ansi": "^5.2.0", + "strip-json-comments": "^3.0.1", + "table": "^5.2.3", + "text-table": "^0.2.0", + "v8-compile-cache": "^2.0.3" + }, + "dependencies": { + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, + "requires": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + } + } + }, + "debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "path-key": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", + "dev": true + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true + }, + "shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", + "dev": true, + "requires": { + "shebang-regex": "^1.0.0" + } + }, + "shebang-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", + "dev": true + }, + "which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, + "eslint-plugin-import": { + "version": "2.20.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.20.2.tgz", + "integrity": "sha512-FObidqpXrR8OnCh4iNsxy+WACztJLXAHBO5hK79T1Hc77PgQZkyDGA5Ag9xAvRpglvLNxhH/zSmZ70/pZ31dHg==", + "dev": true, + "requires": { + "array-includes": "^3.0.3", + "array.prototype.flat": "^1.2.1", + "contains-path": "^0.1.0", + "debug": "^2.6.9", + "doctrine": "1.5.0", + "eslint-import-resolver-node": "^0.3.2", + "eslint-module-utils": "^2.4.1", + "has": "^1.0.3", + "minimatch": "^3.0.4", + "object.values": "^1.1.0", + "read-pkg-up": "^2.0.0", + "resolve": "^1.12.0" + } + }, + "eslint-utils": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-1.4.3.tgz", + "integrity": "sha512-fbBN5W2xdY45KulGXmLHZ3c3FHfVYmKg0IrAKGOkT/464PQsx2UeIzfz1RmEci+KLm1bBaAzZAh8+/E+XAeZ8Q==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + } + }, + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + }, + "globals": { + "version": "12.4.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-12.4.0.tgz", + "integrity": "sha512-BWICuzzDvDoH54NHKCseDanAhE3CeDorgDL5MT6LMXXj2WCnd9UC2szdk4AWLfjdgNBCXLUanXYcpBBKOSWGwg==", + "dev": true, + "requires": { + "type-fest": "^0.8.1" + } + }, + "html-escaper": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-3.0.3.tgz", + "integrity": "sha512-RuMffC89BOWQoY0WKGpIhn5gX3iI54O6nRA0yC124NYVtzjmFWBIiFd8M0x+ZdX0P9R4lADg1mgP8C7PxGOWuQ==", + "dev": true + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "regexpp": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-2.0.1.tgz", + "integrity": "sha512-lv0M6+TkDVniA3aD1Eg0DVpfU/booSu7Eev3TDO/mZKHBfVjgCGTV4t4buppESEYDtkArYFOxTJWv6S5C+iaNw==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "type-fest": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", + "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", + "dev": true + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, + "@salesforce/source-deploy-retrieve": { + "version": "5.12.8", + "resolved": "https://registry.npmjs.org/@salesforce/source-deploy-retrieve/-/source-deploy-retrieve-5.12.8.tgz", + "integrity": "sha512-kEl+SKtJczJtZb/xzl10+UrY52v9roWE+W/k/bOBgtAT/1MkWhX5tCmFqjs4yPgGtk9sRIGhsntfgeedNvCPTg==", + "dev": true, + "requires": { + "@salesforce/core": "^2.35.0", + "@salesforce/kit": "^1.5.32", + "@salesforce/ts-types": "^1.4.2", + "archiver": "^5.3.0", + "fast-xml-parser": "^3.17.4", + "graceful-fs": "^4.2.8", + "ignore": "^5.1.8", + "mime": "2.6.0", + "unzipper": "0.10.11", + "xmldom-sfdx-encoding": "^0.1.29" + } + }, + "@salesforce/source-tracking": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/@salesforce/source-tracking/-/source-tracking-1.3.1.tgz", + "integrity": "sha512-I9NVwliT2nP/FlOeYtFFZ896GdldUDxhSGtTa/eopoGm6YQAIwqlnorBR9ypWd4l4x3i8t4fX0pJR4R5I00l9Q==", + "dev": true, + "requires": { + "@salesforce/core": "^2.33.1", + "@salesforce/kit": "^1.5.17", + "@salesforce/source-deploy-retrieve": "^5.9.4", + "graceful-fs": "^4.2.9", + "isomorphic-git": "1.17.0", + "ts-retry-promise": "^0.6.0" + } + }, + "@salesforce/telemetry": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/@salesforce/telemetry/-/telemetry-3.2.0.tgz", + "integrity": "sha512-WMKnuWfZx/iOg/OErJCt4OS72Fp4lru+EFaxPHgZIuhrLxpNOHrqni31QkYmY62eYKr0tLSuJFOs8vkgAcGcrw==", + "dev": true, + "requires": { + "@salesforce/core": "^2.31.0", + "@salesforce/ts-types": "^1.2.1", + "applicationinsights": "^1.4.0", + "axios": "^0.21.2" + } + }, + "@salesforce/templates": { + "version": "54.3.0", + "resolved": "https://registry.npmjs.org/@salesforce/templates/-/templates-54.3.0.tgz", + "integrity": "sha512-MYWjTVj6wgEht6UWvPgl1v8FY47zAAqAyc/VjdZ4PCwrxGm1RYOyep15B4pF1dkBELD3eses2McN5qKjUPIruw==", + "dev": true, + "requires": { + "@salesforce/core": "^2.33.1", + "got": "^11.8.2", + "mime-types": "^2.1.27", + "tar": "^6.1.10", + "tslib": "^1", + "yeoman-environment": "2.4.0", + "yeoman-generator": "^4.8.2" + }, + "dependencies": { + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true + }, + "ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true + }, + "cli-cursor": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", + "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", + "dev": true, + "requires": { + "restore-cursor": "^2.0.0" + } + }, + "cli-width": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-2.2.1.tgz", + "integrity": "sha512-GRMWDxpOB6Dgk2E5Uo+3eEBvtOOlimMmpbFiKuLFnQzYDavtLFY3K5ona41jgN/WdRZtG7utuVSVTL4HbZHGkw==", + "dev": true + }, + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", + "dev": true + }, + "dir-glob": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.0.0.tgz", + "integrity": "sha512-37qirFDz8cA5fimp9feo43fSuRo2gHwaIn6dXL8Ber1dGwUosDrGZeCCXq57WnIqE4aQ+u3eQZzsk1yOzhdwag==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "path-type": "^3.0.0" + } + }, + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, + "figures": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-2.0.0.tgz", + "integrity": "sha1-OrGi0qYsi/tDGgyUy3l6L84nyWI=", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "requires": { + "minipass": "^3.0.0" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/globby/-/globby-8.0.2.tgz", + "integrity": "sha512-yTzMmKygLp8RUpG1Ymu2VXPSJQZjNAZPD4ywgYEaG7e4tBJeUQBO8OpXrf1RCNcEs5alsoJYPAMiIHP0cmeC7w==", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "dir-glob": "2.0.0", + "fast-glob": "^2.0.2", + "glob": "^7.1.2", + "ignore": "^3.3.5", + "pify": "^3.0.0", + "slash": "^1.0.0" + } + }, + "got": { + "version": "11.8.3", + "resolved": "https://registry.npmjs.org/got/-/got-11.8.3.tgz", + "integrity": "sha512-7gtQ5KiPh1RtGS9/Jbv1ofDpBFuq42gyfEib+ejaRBJuj/3tQFeR5+gw57e4ipaU8c/rCjvX6fkQz2lyDlGAOg==", + "dev": true, + "requires": { + "@sindresorhus/is": "^4.0.0", + "@szmarczak/http-timer": "^4.0.5", + "@types/cacheable-request": "^6.0.1", + "@types/responselike": "^1.0.0", + "cacheable-lookup": "^5.0.3", + "cacheable-request": "^7.0.2", + "decompress-response": "^6.0.0", + "http2-wrapper": "^1.0.0-beta.5.2", + "lowercase-keys": "^2.0.0", + "p-cancelable": "^2.0.0", + "responselike": "^2.0.0" + } + }, + "grouped-queue": { + "version": "0.3.3", + "resolved": "https://registry.npmjs.org/grouped-queue/-/grouped-queue-0.3.3.tgz", + "integrity": "sha1-wWfSpTGcWg4JZO9qJbfC34mWyFw=", + "dev": true, + "requires": { + "lodash": "^4.17.2" + } + }, + "ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "inquirer": { + "version": "6.5.2", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-6.5.2.tgz", + "integrity": "sha512-cntlB5ghuB0iuO65Ovoi8ogLHiWGs/5yNrtUcKjFhSSiVeAIVpD7koaSU9RM8mpXw5YDi9RdYXGQMaOURB7ycQ==", + "dev": true, + "requires": { + "ansi-escapes": "^3.2.0", + "chalk": "^2.4.2", + "cli-cursor": "^2.1.0", + "cli-width": "^2.0.0", + "external-editor": "^3.0.3", + "figures": "^2.0.0", + "lodash": "^4.17.12", + "mute-stream": "0.0.7", + "run-async": "^2.2.0", + "rxjs": "^6.4.0", + "string-width": "^2.1.0", + "strip-ansi": "^5.1.0", + "through": "^2.3.6" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + } + } + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "lowercase-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", + "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", + "dev": true + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + } + }, + "mimic-fn": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", + "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", + "dev": true + }, + "minipass": { + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.6.tgz", + "integrity": "sha512-rty5kpw9/z8SX9dmxblFA6edItUmwJgMeYDZRrwlIVN27i8gysGbznJwUggw2V/FVqFSDdWy040ZPS811DYAqQ==", + "dev": true, + "requires": { + "yallist": "^4.0.0" + } + }, + "minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "requires": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + } + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "mute-stream": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.7.tgz", + "integrity": "sha1-MHXOk7whuPq0PhvE2n6BFe0ee6s=", + "dev": true + }, + "onetime": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", + "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", + "dev": true, + "requires": { + "mimic-fn": "^1.0.0" + } + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + } + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "restore-cursor": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", + "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", + "dev": true, + "requires": { + "onetime": "^2.0.0", + "signal-exit": "^3.0.2" + } + }, + "slash": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", + "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=", + "dev": true + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + } + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + }, + "tar": { + "version": "6.1.11", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.1.11.tgz", + "integrity": "sha512-an/KZQzQUkZCkuoAA64hM92X0Urb6VpRhAFllDzz44U2mcD5scmT3zBc4VgVpkugF580+DQn8eAFSyoQt0tznA==", + "dev": true, + "requires": { + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "minipass": "^3.0.0", + "minizlib": "^2.1.1", + "mkdirp": "^1.0.3", + "yallist": "^4.0.0" + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + } + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "untildify": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/untildify/-/untildify-3.0.3.tgz", + "integrity": "sha512-iSk/J8efr8uPT/Z4eSUywnqyrQU7DSdMfdqK4iWEaUVVmcP5JcnpRqmVMwcwcnmI1ATFNgC5V90u09tBynNFKA==", + "dev": true + }, + "yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "yeoman-environment": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/yeoman-environment/-/yeoman-environment-2.4.0.tgz", + "integrity": "sha512-SsvoL0RNAFIX69eFxkUhwKUN2hG1UwUjxrcP+T2ytwdhqC/kHdnFOH2SXdtSN1Ju4aO4xuimmzfRoheYY88RuA==", + "dev": true, + "requires": { + "chalk": "^2.4.1", + "cross-spawn": "^6.0.5", + "debug": "^3.1.0", + "diff": "^3.5.0", + "escape-string-regexp": "^1.0.2", + "globby": "^8.0.1", + "grouped-queue": "^0.3.3", + "inquirer": "^6.0.0", + "is-scoped": "^1.0.0", + "lodash": "^4.17.10", + "log-symbols": "^2.2.0", + "mem-fs": "^1.1.0", + "strip-ansi": "^4.0.0", + "text-table": "^0.2.0", + "untildify": "^3.0.3" + } + } + } + }, + "@salesforce/ts-types": { + "version": "1.5.20", + "resolved": "https://registry.npmjs.org/@salesforce/ts-types/-/ts-types-1.5.20.tgz", + "integrity": "sha512-Ov6um4CWd63EvkRavkHG0J/P9XYL55sdkDWPMr7+AIgqh5flHxDRz09/C4e9M94aX30rzJxW4TVX6EBf4Cu2BQ==", + "dev": true, + "requires": { + "tslib": "^2.2.0" + } + }, + "@salesforce/wire-service-jest-util": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@salesforce/wire-service-jest-util/-/wire-service-jest-util-4.0.0.tgz", + "integrity": "sha512-YJO/bMq5l6IYIZG6bAqYzzbmZMPCzB2GE2TKGLA1U7B9HpmNHZS7DdDcy154P03dfLSgF+tgVYeklh2HRYGk9g==", + "dev": true + }, + "@sideway/address": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/@sideway/address/-/address-4.1.4.tgz", + "integrity": "sha512-7vwq+rOHVWjyXxVlR76Agnvhy8I9rpzjosTESvmhNeXOXdZZB15Fl+TI9x1SiHZH5Jv2wTGduSxFDIaq0m3DUw==", + "dev": true, + "requires": { + "@hapi/hoek": "^9.0.0" + } + }, + "@sideway/formula": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@sideway/formula/-/formula-3.0.0.tgz", + "integrity": "sha512-vHe7wZ4NOXVfkoRb8T5otiENVlT7a3IAiw7H5M2+GO+9CDgcVUUsX1zalAztCmwyOr2RUTGJdgB+ZvSVqmdHmg==", + "dev": true + }, + "@sideway/pinpoint": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/@sideway/pinpoint/-/pinpoint-2.0.0.tgz", + "integrity": "sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ==", + "dev": true + }, + "@sindresorhus/is": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-4.6.0.tgz", + "integrity": "sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw==", + "dev": true + }, + "@sinonjs/commons": { + "version": "1.8.3", + "resolved": "https://registry.npmjs.org/@sinonjs/commons/-/commons-1.8.3.tgz", + "integrity": "sha512-xkNcLAn/wZaX14RPlwizcKicDk9G3F8m2nU3L7Ukm5zBgTwiT0wsoFAHx9Jq56fJA1z/7uKGtCRu16sOUCLIHQ==", + "dev": true, + "requires": { + "type-detect": "4.0.8" + } + }, + "@sinonjs/fake-timers": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/@sinonjs/fake-timers/-/fake-timers-8.1.0.tgz", + "integrity": "sha512-OAPJUAtgeINhh/TAlUID4QTs53Njm7xzddaVlEs/SXwgtiD1tW22zAB/W1wdqfrpmikgaWQ9Fw6Ws+hsiRm5Vg==", + "dev": true, + "requires": { + "@sinonjs/commons": "^1.7.0" + } + }, + "@szmarczak/http-timer": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/@szmarczak/http-timer/-/http-timer-4.0.6.tgz", + "integrity": "sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w==", + "dev": true, + "requires": { + "defer-to-connect": "^2.0.0" + } + }, + "@tootallnate/once": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", + "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", + "dev": true + }, + "@types/babel__core": { + "version": "7.1.19", + "resolved": "https://registry.npmjs.org/@types/babel__core/-/babel__core-7.1.19.tgz", + "integrity": "sha512-WEOTgRsbYkvA/KCsDwVEGkd7WAr1e3g31VHQ8zy5gul/V1qKullU/BU5I68X5v7V3GnB9eotmom4v5a5gjxorw==", + "dev": true, + "requires": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0", + "@types/babel__generator": "*", + "@types/babel__template": "*", + "@types/babel__traverse": "*" + } + }, + "@types/babel__generator": { + "version": "7.6.4", + "resolved": "https://registry.npmjs.org/@types/babel__generator/-/babel__generator-7.6.4.tgz", + "integrity": "sha512-tFkciB9j2K755yrTALxD44McOrk+gfpIpvC3sxHjRawj6PfnQxrse4Clq5y/Rq+G3mrBurMax/lG8Qn2t9mSsg==", + "dev": true, + "requires": { + "@babel/types": "^7.0.0" + } + }, + "@types/babel__template": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/@types/babel__template/-/babel__template-7.4.1.tgz", + "integrity": "sha512-azBFKemX6kMg5Io+/rdGT0dkGreboUVR0Cdm3fz9QJWpaQGJRQXl7C+6hOTCZcMll7KFyEQpgbYI2lHdsS4U7g==", + "dev": true, + "requires": { + "@babel/parser": "^7.1.0", + "@babel/types": "^7.0.0" + } + }, + "@types/babel__traverse": { + "version": "7.14.2", + "resolved": "https://registry.npmjs.org/@types/babel__traverse/-/babel__traverse-7.14.2.tgz", + "integrity": "sha512-K2waXdXBi2302XUdcHcR1jCeU0LL4TD9HRs/gk0N2Xvrht+G/BfJa4QObBQZfhMdxiCpV3COl5Nfq4uKTeTnJA==", + "dev": true, + "requires": { + "@babel/types": "^7.3.0" + } + }, + "@types/cacheable-request": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/@types/cacheable-request/-/cacheable-request-6.0.2.tgz", + "integrity": "sha512-B3xVo+dlKM6nnKTcmm5ZtY/OL8bOAOd2Olee9M1zft65ox50OzjEHW91sDiU9j6cvW8Ejg1/Qkf4xd2kugApUA==", + "dev": true, + "requires": { + "@types/http-cache-semantics": "*", + "@types/keyv": "*", + "@types/node": "*", + "@types/responselike": "*" + } + }, + "@types/chai": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/@types/chai/-/chai-4.3.0.tgz", + "integrity": "sha512-/ceqdqeRraGolFTcfoXNiqjyQhZzbINDngeoAq9GoHa8PPK1yNzTaxWjA6BFWp5Ua9JpXEMSS4s5i9tS0hOJtw==", + "dev": true + }, + "@types/eslint-visitor-keys": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@types/eslint-visitor-keys/-/eslint-visitor-keys-1.0.0.tgz", + "integrity": "sha512-OCutwjDZ4aFS6PB1UZ988C4YgwlBHJd6wCeQqaLdmadZ/7e+w79+hbMUFC1QXDNCmdyoRfAFdm0RypzwR+Qpag==", + "dev": true + }, + "@types/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", + "dev": true, + "requires": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "@types/graceful-fs": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/@types/graceful-fs/-/graceful-fs-4.1.5.tgz", + "integrity": "sha512-anKkLmZZ+xm4p8JWBf4hElkM4XR+EZeA2M9BAkkTldmcyDY4mbdIJnRghDJH3Ov5ooY7/UAoENtmdMSkaAd7Cw==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/http-cache-semantics": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/@types/http-cache-semantics/-/http-cache-semantics-4.0.1.tgz", + "integrity": "sha512-SZs7ekbP8CN0txVG2xVRH6EgKmEm31BOxA07vkFaETzZz1xh+cbt8BcI0slpymvwhx5dlFnQG2rTlPVQn+iRPQ==", + "dev": true + }, + "@types/istanbul-lib-coverage": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.4.tgz", + "integrity": "sha512-z/QT1XN4K4KYuslS23k62yDIDLwLFkzxOuMplDtObz0+y7VqJCaO2o+SPwHCvLFZh7xazvvoor2tA/hPz9ee7g==", + "dev": true + }, + "@types/istanbul-lib-report": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", + "integrity": "sha512-plGgXAPfVKFoYfa9NpYDAkseG+g6Jr294RqeqcqDixSbU34MZVJRi/P+7Y8GDpzkEwLaGZZOpKIEmeVZNtKsrg==", + "dev": true, + "requires": { + "@types/istanbul-lib-coverage": "*" + } + }, + "@types/istanbul-reports": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/@types/istanbul-reports/-/istanbul-reports-3.0.1.tgz", + "integrity": "sha512-c3mAZEuK0lvBp8tmuL74XRKn1+y2dcwOUpH7x4WrF6gk1GIgiluDRgMYQtw2OFcBvAJWlt6ASU3tSqxp0Uu0Aw==", + "dev": true, + "requires": { + "@types/istanbul-lib-report": "*" + } + }, + "@types/jsforce": { + "version": "1.9.41", + "resolved": "https://registry.npmjs.org/@types/jsforce/-/jsforce-1.9.41.tgz", + "integrity": "sha512-J0dReK6EPGR98b4fAowqqQqFXH4DGtPxY2lLrZGcuCthrHYkYNrKnfGf2xM1jwiBC5CGdSEDmWEDwRwwmX25tA==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/json-buffer": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/json-buffer/-/json-buffer-3.0.0.tgz", + "integrity": "sha512-3YP80IxxFJB4b5tYC2SUPwkg0XQLiu0nWvhRgEatgjf+29IcWO9X1k8xRv5DGssJ/lCrjYTjQPcobJr2yWIVuQ==", + "dev": true + }, + "@types/json-schema": { + "version": "7.0.11", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.11.tgz", + "integrity": "sha512-wOuvG1SN4Us4rez+tylwwwCV1psiNVOkJeM3AUWUNWg/jDQY2+HE/444y5gc+jBmRqASOm2Oeh5c1axHobwRKQ==", + "dev": true + }, + "@types/json5": { + "version": "0.0.29", + "resolved": "https://registry.npmjs.org/@types/json5/-/json5-0.0.29.tgz", + "integrity": "sha1-7ihweulOEdK4J7y+UnC86n8+ce4=", + "dev": true + }, + "@types/keyv": { + "version": "3.1.4", + "resolved": "https://registry.npmjs.org/@types/keyv/-/keyv-3.1.4.tgz", + "integrity": "sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/linkify-it": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/@types/linkify-it/-/linkify-it-3.0.2.tgz", + "integrity": "sha512-HZQYqbiFVWufzCwexrvh694SOim8z2d+xJl5UNamcvQFejLY/2YUtzXHYi3cHdI7PMlS8ejH2slRAOJQ32aNbA==", + "dev": true + }, + "@types/lodash": { + "version": "4.14.181", + "resolved": "https://registry.npmjs.org/@types/lodash/-/lodash-4.14.181.tgz", + "integrity": "sha512-n3tyKthHJbkiWhDZs3DkhkCzt2MexYHXlX0td5iMplyfwketaOeKboEVBqzceH7juqvEg3q5oUoBFxSLu7zFag==", + "dev": true + }, + "@types/markdown-it": { + "version": "12.2.3", + "resolved": "https://registry.npmjs.org/@types/markdown-it/-/markdown-it-12.2.3.tgz", + "integrity": "sha512-GKMHFfv3458yYy+v/N8gjufHO6MSZKCOXpZc5GXIWWy8uldwfmPn98vp81gZ5f9SVw8YYBctgfJ22a2d7AOMeQ==", + "dev": true, + "requires": { + "@types/linkify-it": "*", + "@types/mdurl": "*" + } + }, + "@types/mdurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@types/mdurl/-/mdurl-1.0.2.tgz", + "integrity": "sha512-eC4U9MlIcu2q0KQmXszyn5Akca/0jrQmwDRgpAMJai7qBWq4amIQhZyNau4VYGtCeALvW1/NtjzJJ567aZxfKA==", + "dev": true + }, + "@types/minimatch": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", + "dev": true + }, + "@types/mkdirp": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@types/mkdirp/-/mkdirp-1.0.2.tgz", + "integrity": "sha512-o0K1tSO0Dx5X6xlU5F1D6625FawhC3dU3iqr25lluNv/+/QIVH8RLNEiVokgIZo+mz+87w/3Mkg/VvQS+J51fQ==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/node": { + "version": "17.0.23", + "resolved": "https://registry.npmjs.org/@types/node/-/node-17.0.23.tgz", + "integrity": "sha512-UxDxWn7dl97rKVeVS61vErvw086aCYhDLyvRQZ5Rk65rZKepaFdm53GeqXaKBuOhED4e9uWq34IC3TdSdJJ2Gw==", + "dev": true + }, + "@types/normalize-package-data": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.1.tgz", + "integrity": "sha512-Gj7cI7z+98M282Tqmp2K5EIsoouUEzbBJhQQzDE3jSIRk6r9gsz0oUokqIUR4u1R3dMHo0pDHM7sNOHyhulypw==", + "dev": true + }, + "@types/prettier": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/@types/prettier/-/prettier-2.6.0.tgz", + "integrity": "sha512-G/AdOadiZhnJp0jXCaBQU449W2h716OW/EoXeYkCytxKL06X1WCXB4DZpp8TpZ8eyIJVS1cw4lrlkkSYU21cDw==", + "dev": true + }, + "@types/responselike": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/@types/responselike/-/responselike-1.0.0.tgz", + "integrity": "sha512-85Y2BjiufFzaMIlvJDvTTB8Fxl2xfLo4HgmHzVBz08w4wDePCTjYw66PdrolO0kzli3yam/YCgRufyo1DdQVTA==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/semver": { + "version": "7.3.9", + "resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.3.9.tgz", + "integrity": "sha512-L/TMpyURfBkf+o/526Zb6kd/tchUP3iBDEPjqjb+U2MAJhVRxxrmr2fwpe08E7QsV7YLcpq0tUaQ9O9x97ZIxQ==", + "dev": true + }, + "@types/sinon": { + "version": "10.0.11", + "resolved": "https://registry.npmjs.org/@types/sinon/-/sinon-10.0.11.tgz", + "integrity": "sha512-dmZsHlBsKUtBpHriNjlK0ndlvEh8dcb9uV9Afsbt89QIyydpC7NcR+nWlAhASfy3GHnxTl4FX/aKE7XZUt/B4g==", + "dev": true, + "requires": { + "@types/sinonjs__fake-timers": "*" + } + }, + "@types/sinonjs__fake-timers": { + "version": "8.1.2", + "resolved": "https://registry.npmjs.org/@types/sinonjs__fake-timers/-/sinonjs__fake-timers-8.1.2.tgz", + "integrity": "sha512-9GcLXF0/v3t80caGs5p2rRfkB+a8VBGLJZVih6CNFkx8IZ994wiKKLSRs9nuFwk1HevWs/1mnUmkApGrSGsShA==", + "dev": true + }, + "@types/stack-utils": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/@types/stack-utils/-/stack-utils-2.0.1.tgz", + "integrity": "sha512-Hl219/BT5fLAaz6NDkSuhzasy49dwQS/DSdu4MdggFB8zcXv7vflBI3xp7FEmkmdDkBUI2bPUNeMttp2knYdxw==", + "dev": true + }, + "@types/yargs": { + "version": "16.0.4", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-16.0.4.tgz", + "integrity": "sha512-T8Yc9wt/5LbJyCaLiHPReJa0kApcIgJ7Bn735GjItUfh08Z1pJvu8QZqb9s+mMvKV6WUQRV7K2R46YbjMXTTJw==", + "dev": true, + "requires": { + "@types/yargs-parser": "*" + } + }, + "@types/yargs-parser": { + "version": "21.0.0", + "resolved": "https://registry.npmjs.org/@types/yargs-parser/-/yargs-parser-21.0.0.tgz", + "integrity": "sha512-iO9ZQHkZxHn4mSakYV0vFHAVDyEOIJQrV2uZ06HxEPcx+mt8swXoZHIbaaJ2crJYFfErySgktuTZ3BeLz+XmFA==", + "dev": true + }, + "@typescript-eslint/eslint-plugin": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-2.34.0.tgz", + "integrity": "sha512-4zY3Z88rEE99+CNvTbXSyovv2z9PNOVffTWD2W8QF5s2prBQtwN2zadqERcrHpcR7O/+KMI3fcTAmUUhK/iQcQ==", + "dev": true, + "requires": { + "@typescript-eslint/experimental-utils": "2.34.0", + "functional-red-black-tree": "^1.0.1", + "regexpp": "^3.0.0", + "tsutils": "^3.17.1" + } + }, + "@typescript-eslint/experimental-utils": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/experimental-utils/-/experimental-utils-2.34.0.tgz", + "integrity": "sha512-eS6FTkq+wuMJ+sgtuNTtcqavWXqsflWcfBnlYhg/nS4aZ1leewkXGbvBhaapn1q6qf4M71bsR1tez5JTRMuqwA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.3", + "@typescript-eslint/typescript-estree": "2.34.0", + "eslint-scope": "^5.0.0", + "eslint-utils": "^2.0.0" + } + }, + "@typescript-eslint/parser": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-2.34.0.tgz", + "integrity": "sha512-03ilO0ucSD0EPTw2X4PntSIRFtDPWjrVq7C3/Z3VQHRC7+13YB55rcJI3Jt+YgeHbjUdJPcPa7b23rXCBokuyA==", + "dev": true, + "requires": { + "@types/eslint-visitor-keys": "^1.0.0", + "@typescript-eslint/experimental-utils": "2.34.0", + "@typescript-eslint/typescript-estree": "2.34.0", + "eslint-visitor-keys": "^1.1.0" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + } + } + }, + "@typescript-eslint/typescript-estree": { + "version": "2.34.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-2.34.0.tgz", + "integrity": "sha512-OMAr+nJWKdlVM9LOqCqh3pQQPwxHAN7Du8DR6dmwCrAmxtiXQnhHJ6tBNtf+cggqfo51SG/FCwnKhXCIM7hnVg==", + "dev": true, + "requires": { + "debug": "^4.1.1", + "eslint-visitor-keys": "^1.1.0", + "glob": "^7.1.6", + "is-glob": "^4.0.1", + "lodash": "^4.17.15", + "semver": "^7.3.2", + "tsutils": "^3.17.1" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } + } + }, + "@xml-tools/parser": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@xml-tools/parser/-/parser-1.0.11.tgz", + "integrity": "sha512-aKqQ077XnR+oQtHJlrAflaZaL7qZsulWc/i/ZEooar5JiWj1eLt0+Wg28cpa+XLney107wXqneC+oG1IZvxkTA==", + "dev": true, + "requires": { + "chevrotain": "7.1.1" + } + }, + "JSONStream": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.5.tgz", + "integrity": "sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==", + "dev": true, + "requires": { + "jsonparse": "^1.2.0", + "through": ">=2.2.7 <3" + } + }, + "abab": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.5.tgz", + "integrity": "sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q==", + "dev": true + }, + "abort-controller": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/abort-controller/-/abort-controller-3.0.0.tgz", + "integrity": "sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==", + "dev": true, + "requires": { + "event-target-shim": "^5.0.0" + } + }, + "acorn": { + "version": "8.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.4.1.tgz", + "integrity": "sha512-asabaBSkEKosYKMITunzX177CXxQ4Q8BSSzMTKD+FefUhipQC70gfW5SiUDhYQ3vk8G+81HqQk7Fv9OXwwn9KA==", + "dev": true + }, + "acorn-globals": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-6.0.0.tgz", + "integrity": "sha512-ZQl7LOWaF5ePqqcX4hLuv/bLXYQNfNWw2c0/yX/TsPRKamzHcTGQnlCjHT3TsmkOUVEPS3crCxiPfdzE/Trlhg==", + "dev": true, + "requires": { + "acorn": "^7.1.1", + "acorn-walk": "^7.1.1" + }, + "dependencies": { + "acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "dev": true + } + } + }, + "acorn-jsx": { + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/acorn-jsx/-/acorn-jsx-5.3.2.tgz", + "integrity": "sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ==", + "dev": true + }, + "acorn-walk": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-7.2.0.tgz", + "integrity": "sha512-OPdCF6GsMIP+Az+aWfAAOEt2/+iVDKE7oy6lJ098aoe59oAmK76qV6Gw60SbZ8jHuG2wH058GF4pLFbYamYrVA==", + "dev": true + }, + "adm-zip": { + "version": "0.5.9", + "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.5.9.tgz", + "integrity": "sha512-s+3fXLkeeLjZ2kLjCBwQufpI5fuN+kIGBxu6530nVQZGVol0d7Y/M88/xw9HGGUcJjKf8LutN3VPRUBq6N7Ajg==", + "dev": true + }, + "agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dev": true, + "requires": { + "debug": "4" + } + }, + "aggregate-error": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", + "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", + "dev": true, + "requires": { + "clean-stack": "^2.0.0", + "indent-string": "^4.0.0" + }, + "dependencies": { + "clean-stack": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", + "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", + "dev": true + } + } + }, + "ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "alce": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/alce/-/alce-1.2.0.tgz", + "integrity": "sha1-qL4trKrEJJRhLxjcCdtpHz3qSqs=", + "dev": true, + "requires": { + "esprima": "^1.2.0", + "estraverse": "^1.5.0" + }, + "dependencies": { + "esprima": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-1.2.5.tgz", + "integrity": "sha1-CZNQL+r2aBODJXVvMPmlH+7sEek=", + "dev": true + }, + "estraverse": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-1.9.3.tgz", + "integrity": "sha1-r2fy3JIlgkFZUJJgkaQAXSnJu0Q=", + "dev": true + } + } + }, + "ansi-escape-sequences": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-escape-sequences/-/ansi-escape-sequences-4.1.0.tgz", + "integrity": "sha512-dzW9kHxH011uBsidTXd14JXgzye/YLb2LzeKZ4bsgl/Knwx8AtbSFkkGxagdNOoh0DlqHCmfiEjWKBaqjOanVw==", + "dev": true, + "requires": { + "array-back": "^3.0.1" + }, + "dependencies": { + "array-back": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz", + "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==", + "dev": true + } + } + }, + "ansi-escapes": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", + "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", + "dev": true + }, + "ansi-regex": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dev": true + }, + "ansi-styles": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", + "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", + "dev": true, + "requires": { + "color-convert": "^1.9.0" + } + }, + "ansicolors": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/ansicolors/-/ansicolors-0.3.2.tgz", + "integrity": "sha1-ZlWX3oap/+Oqm/vmyuXG6kJrSXk=", + "dev": true + }, + "anymatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", + "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", + "dev": true, + "requires": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + } + }, + "applicationinsights": { + "version": "1.8.10", + "resolved": "https://registry.npmjs.org/applicationinsights/-/applicationinsights-1.8.10.tgz", + "integrity": "sha512-ZLDA7mShh4mP2Z/HlFolmvhBPX1LfnbIWXrselyYVA7EKjHhri1fZzpu2EiWAmfbRxNBY6fRjoPJWbx5giKy4A==", + "dev": true, + "requires": { + "cls-hooked": "^4.2.2", + "continuation-local-storage": "^3.2.1", + "diagnostic-channel": "0.3.1", + "diagnostic-channel-publishers": "0.4.4" + } + }, + "archiver": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/archiver/-/archiver-5.3.0.tgz", + "integrity": "sha512-iUw+oDwK0fgNpvveEsdQ0Ase6IIKztBJU2U0E9MzszMfmVVUyv1QJhS2ITW9ZCqx8dktAxVAjWWkKehuZE8OPg==", + "dev": true, + "requires": { + "archiver-utils": "^2.1.0", + "async": "^3.2.0", + "buffer-crc32": "^0.2.1", + "readable-stream": "^3.6.0", + "readdir-glob": "^1.0.0", + "tar-stream": "^2.2.0", + "zip-stream": "^4.1.0" + } + }, + "archiver-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/archiver-utils/-/archiver-utils-2.1.0.tgz", + "integrity": "sha512-bEL/yUb/fNNiNTuUz979Z0Yg5L+LzLxGJz8x79lYmR54fmTIb6ob/hNQgkQnIUDWIFjZVQwl9Xs356I6BAMHfw==", + "dev": true, + "requires": { + "glob": "^7.1.4", + "graceful-fs": "^4.2.0", + "lazystream": "^1.0.0", + "lodash.defaults": "^4.2.0", + "lodash.difference": "^4.5.0", + "lodash.flatten": "^4.4.0", + "lodash.isplainobject": "^4.0.6", + "lodash.union": "^4.6.0", + "normalize-path": "^3.0.0", + "readable-stream": "^2.0.0" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } + } + }, + "arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true + }, + "argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "requires": { + "sprintf-js": "~1.0.2" + } + }, + "arr-diff": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", + "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", + "dev": true + }, + "arr-flatten": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", + "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", + "dev": true + }, + "arr-union": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", + "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", + "dev": true + }, + "array-back": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-6.2.2.tgz", + "integrity": "sha512-gUAZ7HPyb4SJczXAMUXMGAvI976JoK3qEx9v1FTmeYuJj0IBiaKttG1ydtGKdkfqWkIkouke7nG8ufGy77+Cvw==", + "dev": true + }, + "array-differ": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/array-differ/-/array-differ-3.0.0.tgz", + "integrity": "sha512-THtfYS6KtME/yIAhKjZ2ul7XI96lQGHRputJQHO80LAWQnuGP4iCIN8vdMRboGbIEYBwU33q8Tch1os2+X0kMg==", + "dev": true + }, + "array-includes": { + "version": "3.1.4", + "resolved": "https://registry.npmjs.org/array-includes/-/array-includes-3.1.4.tgz", + "integrity": "sha512-ZTNSQkmWumEbiHO2GF4GmWxYVTiQyJy2XOTa15sdQSrvKn7l+180egQMqlrMOUMCyLMD7pmyQe4mMDUT6Behrw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.1", + "get-intrinsic": "^1.1.1", + "is-string": "^1.0.7" + } + }, + "array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true + }, + "array-uniq": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", + "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", + "dev": true + }, + "array-unique": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", + "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", + "dev": true + }, + "array.prototype.flat": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/array.prototype.flat/-/array.prototype.flat-1.3.0.tgz", + "integrity": "sha512-12IUEkHsAhA4DY5s0FPgNXIdc8VRSqD9Zp78a5au9abH/SOBrsp082JOWFNTjkMozh8mqcdiKuaLGhPeYztxSw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3", + "es-abstract": "^1.19.2", + "es-shim-unscopables": "^1.0.0" + } + }, + "arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": true + }, + "asap": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", + "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=", + "dev": true + }, + "asn1": { + "version": "0.2.6", + "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.6.tgz", + "integrity": "sha512-ix/FxPn0MDjeyJ7i/yoHGFt/EX6LyNbxSEhPPXODPL+KB0VPk86UYfL0lMdy+KCnv+fmvIzySwaK5COwqVbWTQ==", + "dev": true, + "requires": { + "safer-buffer": "~2.1.0" + } + }, + "assert-plus": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", + "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", + "dev": true + }, + "assign-symbols": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", + "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", + "dev": true + }, + "ast-metadata-inferer": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/ast-metadata-inferer/-/ast-metadata-inferer-0.7.0.tgz", + "integrity": "sha512-OkMLzd8xelb3gmnp6ToFvvsHLtS6CbagTkFQvQ+ZYFe3/AIl9iKikNR9G7pY3GfOR/2Xc222hwBjzI7HLkE76Q==", + "dev": true, + "requires": { + "@mdn/browser-compat-data": "^3.3.14" + }, + "dependencies": { + "@mdn/browser-compat-data": { + "version": "3.3.14", + "resolved": "https://registry.npmjs.org/@mdn/browser-compat-data/-/browser-compat-data-3.3.14.tgz", + "integrity": "sha512-n2RC9d6XatVbWFdHLimzzUJxJ1KY8LdjqrW6YvGPiRmsHkhOUx74/Ct10x5Yo7bC/Jvqx7cDEW8IMPv/+vwEzA==", + "dev": true + } + } + }, + "ast-types": { + "version": "0.13.4", + "resolved": "https://registry.npmjs.org/ast-types/-/ast-types-0.13.4.tgz", + "integrity": "sha512-x1FCFnFifvYDDzTaLII71vG5uvDwgtmDTEVWAxrgeiR8VjMONcCXJx7E+USjDtHlwFmt9MysbqgF9b9Vjr6w+w==", + "dev": true, + "requires": { + "tslib": "^2.0.1" + } + }, + "astral-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-1.0.0.tgz", + "integrity": "sha512-+Ryf6g3BKoRc7jfp7ad8tM4TtMiaWvbF/1/sQcZPkkS7ag3D5nMBCe2UfOTONtAkaG0tO0ij3C5Lwmf1EiyjHg==", + "dev": true + }, + "astring": { + "version": "1.7.6", + "resolved": "https://registry.npmjs.org/astring/-/astring-1.7.6.tgz", + "integrity": "sha512-7ofuRb7zx2u7T4OGZTtfkGKAfPKq72XQ7zgpI2b3pR3wdROrDIDmKPtrel7D8S4t+97SGpYTpDR6lq5cNX/DJw==", + "dev": true + }, + "async": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.3.tgz", + "integrity": "sha512-spZRyzKL5l5BZQrr/6m/SqFdBN0q3OCI0f9rjfBzCMBIP4p75P620rR3gTmaksNOhmzgdxcaxdNfMy6anrbM0g==", + "dev": true + }, + "async-hook-jl": { + "version": "1.7.6", + "resolved": "https://registry.npmjs.org/async-hook-jl/-/async-hook-jl-1.7.6.tgz", + "integrity": "sha512-gFaHkFfSxTjvoxDMYqDuGHlcRyUuamF8s+ZTtJdDzqjws4mCt7v0vuV79/E2Wr2/riMQgtG4/yUtXWs1gZ7JMg==", + "dev": true, + "requires": { + "stack-chain": "^1.3.7" + } + }, + "async-listener": { + "version": "0.6.10", + "resolved": "https://registry.npmjs.org/async-listener/-/async-listener-0.6.10.tgz", + "integrity": "sha512-gpuo6xOyF4D5DE5WvyqZdPA3NGhiT6Qf07l7DCB0wwDEsLvDIbCr6j9S5aj5Ch96dLace5tXVzWBZkxU/c5ohw==", + "dev": true, + "requires": { + "semver": "^5.3.0", + "shimmer": "^1.1.0" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + } + } + }, + "async-lock": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/async-lock/-/async-lock-1.3.1.tgz", + "integrity": "sha512-zK7xap9UnttfbE23JmcrNIyueAn6jWshihJqA33U/hEnKprF/lVGBDsBv/bqLm2YMMl1DnpHhUY044eA0t1TUw==", + "dev": true + }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, + "at-least-node": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/at-least-node/-/at-least-node-1.0.0.tgz", + "integrity": "sha512-+q/t7Ekv1EDY2l6Gda6LLiX14rU9TV20Wa3ofeQmwPFZbOMo9DXrLbOjFaaclkXKWidIaopwAObQDqwWtGUjqg==", + "dev": true + }, + "atob": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", + "dev": true + }, + "aws-sign2": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", + "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", + "dev": true + }, + "aws4": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.11.0.tgz", + "integrity": "sha512-xh1Rl34h6Fi1DC2WWKfxUTVqRsNnr6LsKz2+hfwDxQJWmrx8+c7ylaqBMcHfl1U1r2dsifOvKX3LQuLNZ+XSvA==", + "dev": true + }, + "axios": { + "version": "0.21.4", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.21.4.tgz", + "integrity": "sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==", + "dev": true, + "requires": { + "follow-redirects": "^1.14.0" + } + }, + "babel-eslint": { + "version": "10.1.0", + "resolved": "https://registry.npmjs.org/babel-eslint/-/babel-eslint-10.1.0.tgz", + "integrity": "sha512-ifWaTHQ0ce+448CYop8AdrQiBsGrnC+bMgfyKFdi6EsPLTAWG+QfyDeM6OH+FmWnKvEq5NnBMLvlBUPKQZoDSg==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.0.0", + "@babel/parser": "^7.7.0", + "@babel/traverse": "^7.7.0", + "@babel/types": "^7.7.0", + "eslint-visitor-keys": "^1.0.0", + "resolve": "^1.12.0" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + } + } + }, + "babel-jest": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/babel-jest/-/babel-jest-27.5.1.tgz", + "integrity": "sha512-cdQ5dXjGRd0IBRATiQ4mZGlGlRE8kJpjPOixdNRdT+m3UcNqmYWN6rK6nvtXYfY3D76cb8s/O1Ss8ea24PIwcg==", + "dev": true, + "requires": { + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/babel__core": "^7.1.14", + "babel-plugin-istanbul": "^6.1.1", + "babel-preset-jest": "^27.5.1", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "slash": "^3.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "babel-plugin-dynamic-import-node": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz", + "integrity": "sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==", + "dev": true, + "requires": { + "object.assign": "^4.1.0" + } + }, + "babel-plugin-istanbul": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/babel-plugin-istanbul/-/babel-plugin-istanbul-6.1.1.tgz", + "integrity": "sha512-Y1IQok9821cC9onCx5otgFfRm7Lm+I+wwxOx738M/WLPZ9Q42m4IG5W0FNX8WLL2gYMZo3JkuXIH2DOpWM+qwA==", + "dev": true, + "requires": { + "@babel/helper-plugin-utils": "^7.0.0", + "@istanbuljs/load-nyc-config": "^1.0.0", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-instrument": "^5.0.4", + "test-exclude": "^6.0.0" + } + }, + "babel-plugin-jest-hoist": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-27.5.1.tgz", + "integrity": "sha512-50wCwD5EMNW4aRpOwtqzyZHIewTYNxLA4nhB+09d8BIssfNfzBRhkBIHiaPv1Si226TQSvp8gxAJm2iY2qs2hQ==", + "dev": true, + "requires": { + "@babel/template": "^7.3.3", + "@babel/types": "^7.3.3", + "@types/babel__core": "^7.0.0", + "@types/babel__traverse": "^7.0.6" + } + }, + "babel-preset-current-node-syntax": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/babel-preset-current-node-syntax/-/babel-preset-current-node-syntax-1.0.1.tgz", + "integrity": "sha512-M7LQ0bxarkxQoN+vz5aJPsLBn77n8QgTFmo8WK0/44auK2xlCXrYcUxHFxgU7qW5Yzw/CjmLRK2uJzaCd7LvqQ==", + "dev": true, + "requires": { + "@babel/plugin-syntax-async-generators": "^7.8.4", + "@babel/plugin-syntax-bigint": "^7.8.3", + "@babel/plugin-syntax-class-properties": "^7.8.3", + "@babel/plugin-syntax-import-meta": "^7.8.3", + "@babel/plugin-syntax-json-strings": "^7.8.3", + "@babel/plugin-syntax-logical-assignment-operators": "^7.8.3", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3", + "@babel/plugin-syntax-numeric-separator": "^7.8.3", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3", + "@babel/plugin-syntax-optional-chaining": "^7.8.3", + "@babel/plugin-syntax-top-level-await": "^7.8.3" + } + }, + "babel-preset-jest": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/babel-preset-jest/-/babel-preset-jest-27.5.1.tgz", + "integrity": "sha512-Nptf2FzlPCWYuJg41HBqXVT8ym6bXOevuCTbhxlUpjwtysGaIWFvDEjp4y+G7fl13FgOdjs7P/DmErqH7da0Ag==", + "dev": true, + "requires": { + "babel-plugin-jest-hoist": "^27.5.1", + "babel-preset-current-node-syntax": "^1.0.0" + } + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==" + }, + "base": { + "version": "0.11.2", + "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", + "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", + "dev": true, + "requires": { + "cache-base": "^1.0.1", + "class-utils": "^0.3.5", + "component-emitter": "^1.2.1", + "define-property": "^1.0.0", + "isobject": "^3.0.1", + "mixin-deep": "^1.2.0", + "pascalcase": "^0.1.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "base64-js": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz", + "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==", + "dev": true + }, + "base64-url": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/base64-url/-/base64-url-2.3.3.tgz", + "integrity": "sha512-dLMhIsK7OplcDauDH/tZLvK7JmUZK3A7KiQpjNzsBrM6Etw7hzNI1tLEywqJk9NnwkgWuFKSlx/IUO7vF6Mo8Q==", + "dev": true + }, + "base64url": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/base64url/-/base64url-3.0.1.tgz", + "integrity": "sha512-ir1UPr3dkwexU7FdV8qBBbNDRUhMmIekYMFZfi+C/sLNnRESKPl23nB9b2pltqfOQNnGzsDdId90AEtG5tCx4A==", + "dev": true + }, + "bcrypt-pbkdf": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", + "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", + "dev": true, + "requires": { + "tweetnacl": "^0.14.3" + } + }, + "big-integer": { + "version": "1.6.51", + "resolved": "https://registry.npmjs.org/big-integer/-/big-integer-1.6.51.tgz", + "integrity": "sha512-GPEid2Y9QU1Exl1rpO9B2IPJGHPSupF5GnVIP0blYvNOMer2bTvSWs1jGOUg04hTmu67nmLsQ9TBo1puaotBHg==", + "dev": true + }, + "binary": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/binary/-/binary-0.3.0.tgz", + "integrity": "sha1-n2BVO8XOjDOG87VTz/R0Yq3sqnk=", + "dev": true, + "requires": { + "buffers": "~0.1.1", + "chainsaw": "~0.1.0" + } + }, + "binaryextensions": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/binaryextensions/-/binaryextensions-2.3.0.tgz", + "integrity": "sha512-nAihlQsYGyc5Bwq6+EsubvANYGExeJKHDO3RjnvwU042fawQTQfM3Kxn7IHUXQOz4bzfwsGYYHGSvXyW4zOGLg==", + "dev": true + }, + "bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dev": true, + "requires": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "requires": { + "fill-range": "^7.0.1" + } + }, + "browser-process-hrtime": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz", + "integrity": "sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==", + "dev": true + }, + "browserslist": { + "version": "4.20.2", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.20.2.tgz", + "integrity": "sha512-CQOBCqp/9pDvDbx3xfMi+86pr4KXIf2FDkTTdeuYw8OxS9t898LA1Khq57gtufFILXpfgsSx5woNgsBgvGjpsA==", + "dev": true, + "requires": { + "caniuse-lite": "^1.0.30001317", + "electron-to-chromium": "^1.4.84", + "escalade": "^3.1.1", + "node-releases": "^2.0.2", + "picocolors": "^1.0.0" + } + }, + "bser": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/bser/-/bser-2.1.1.tgz", + "integrity": "sha512-gQxTNE/GAfIIrmHLUE3oJyp5FO6HRBfhjnw4/wMmA63ZGDJnWBmgY/lyQBpnDUkGmAhbSe39tx2d/iTOAfglwQ==", + "dev": true, + "requires": { + "node-int64": "^0.4.0" + } + }, + "buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, + "requires": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "buffer-crc32": { + "version": "0.2.13", + "resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-0.2.13.tgz", + "integrity": "sha1-DTM+PwDqxQqhRUq9MO+MKl2ackI=", + "dev": true + }, + "buffer-equal-constant-time": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz", + "integrity": "sha1-+OcRMvf/5uAaXJaXpMbz5I1cyBk=", + "dev": true + }, + "buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "buffer-indexof-polyfill": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/buffer-indexof-polyfill/-/buffer-indexof-polyfill-1.0.2.tgz", + "integrity": "sha512-I7wzHwA3t1/lwXQh+A5PbNvJxgfo5r3xulgpYDB5zckTu/Z9oUK9biouBKQUjEqzaz3HnAT6TYoovmE+GqSf7A==", + "dev": true + }, + "buffers": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/buffers/-/buffers-0.1.1.tgz", + "integrity": "sha1-skV5w77U1tOWru5tmorn9Ugqt7s=", + "dev": true + }, + "byline": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/byline/-/byline-5.0.0.tgz", + "integrity": "sha1-dBxSFkaOrcRXsDQQEYrXfejB3bE=", + "dev": true + }, + "bytes": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", + "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==", + "dev": true + }, + "cache-base": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", + "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", + "dev": true, + "requires": { + "collection-visit": "^1.0.0", + "component-emitter": "^1.2.1", + "get-value": "^2.0.6", + "has-value": "^1.0.0", + "isobject": "^3.0.1", + "set-value": "^2.0.0", + "to-object-path": "^0.3.0", + "union-value": "^1.0.0", + "unset-value": "^1.0.0" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "cache-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/cache-point/-/cache-point-2.0.0.tgz", + "integrity": "sha512-4gkeHlFpSKgm3vm2gJN5sPqfmijYRFYCQ6tv5cLw0xVmT6r1z1vd4FNnpuOREco3cBs1G709sZ72LdgddKvL5w==", + "dev": true, + "requires": { + "array-back": "^4.0.1", + "fs-then-native": "^2.0.0", + "mkdirp2": "^1.0.4" + }, + "dependencies": { + "array-back": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-4.0.2.tgz", + "integrity": "sha512-NbdMezxqf94cnNfWLL7V/im0Ub+Anbb0IoZhvzie8+4HJ4nMQuzHuy49FkGYCJK2yAloZ3meiB6AVMClbrI1vg==", + "dev": true + } + } + }, + "cacheable-lookup": { + "version": "5.0.4", + "resolved": "https://registry.npmjs.org/cacheable-lookup/-/cacheable-lookup-5.0.4.tgz", + "integrity": "sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA==", + "dev": true + }, + "cacheable-request": { + "version": "7.0.2", + "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-7.0.2.tgz", + "integrity": "sha512-pouW8/FmiPQbuGpkXQ9BAPv/Mo5xDGANgSNXzTzJ8DrKGuXOssM4wIQRjfanNRh3Yu5cfYPvcorqbhg2KIJtew==", + "dev": true, + "requires": { + "clone-response": "^1.0.2", + "get-stream": "^5.1.0", + "http-cache-semantics": "^4.0.0", + "keyv": "^4.0.0", + "lowercase-keys": "^2.0.0", + "normalize-url": "^6.0.1", + "responselike": "^2.0.0" + }, + "dependencies": { + "get-stream": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", + "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "lowercase-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", + "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", + "dev": true + } + } + }, + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + } + }, + "call-me-maybe": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.1.tgz", + "integrity": "sha1-JtII6onje1y95gJQoV8DHBak1ms=", + "dev": true + }, + "callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true + }, + "camel-case": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/camel-case/-/camel-case-4.1.2.tgz", + "integrity": "sha512-gxGWBrTT1JuMx6R+o5PTXMmUnhnVzLQ9SNutD4YqKtI6ap897t3tKECYla6gCWEkplXnlNybEkZg9GEGxKFCgw==", + "dev": true, + "requires": { + "pascal-case": "^3.1.2", + "tslib": "^2.0.3" + } + }, + "camelcase": { + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", + "dev": true + }, + "caniuse-lite": { + "version": "1.0.30001331", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001331.tgz", + "integrity": "sha512-Y1xk6paHpUXKP/P6YjQv1xqyTbgAP05ycHBcRdQjTcyXlWol868sJJPlmk5ylOekw2BrucWes5jk+LvVd7WZ5Q==", + "dev": true + }, + "capital-case": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/capital-case/-/capital-case-1.0.4.tgz", + "integrity": "sha512-ds37W8CytHgwnhGGTi88pcPyR15qoNkOpYwmMMfnWqqWgESapLqvDx6huFjQ5vqWSn2Z06173XNA7LtMOeUh1A==", + "dev": true, + "requires": { + "no-case": "^3.0.4", + "tslib": "^2.0.3", + "upper-case-first": "^2.0.2" + } + }, + "capture-stack-trace": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/capture-stack-trace/-/capture-stack-trace-1.0.1.tgz", + "integrity": "sha512-mYQLZnx5Qt1JgB1WEiMCf2647plpGeQ2NMR/5L0HNZzGQo4fuSPnK+wjfPnKZV0aiJDgzmWqqkV/g7JD+DW0qw==", + "dev": true + }, + "cardinal": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/cardinal/-/cardinal-2.1.1.tgz", + "integrity": "sha1-fMEFXYItISlU0HsIXeolHMe8VQU=", + "dev": true, + "requires": { + "ansicolors": "~0.3.2", + "redeyed": "~2.1.0" + } + }, + "caseless": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", + "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", + "dev": true + }, + "catharsis": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/catharsis/-/catharsis-0.9.0.tgz", + "integrity": "sha512-prMTQVpcns/tzFgFVkVp6ak6RykZyWb3gu8ckUpd6YkTlacOd3DXGJjIpD4Q6zJirizvaiAjSSHlOsA+6sNh2A==", + "dev": true, + "requires": { + "lodash": "^4.17.15" + } + }, + "chainsaw": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/chainsaw/-/chainsaw-0.1.0.tgz", + "integrity": "sha1-XqtQsor+WAdNDVgpE4iCi15fvJg=", + "dev": true, + "requires": { + "traverse": ">=0.3.0 <0.4" + } + }, + "chalk": { + "version": "2.4.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", + "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", + "dev": true, + "requires": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + } + }, + "change-case": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/change-case/-/change-case-4.1.2.tgz", + "integrity": "sha512-bSxY2ws9OtviILG1EiY5K7NNxkqg/JnRnFxLtKQ96JaviiIxi7djMrSd0ECT9AC+lttClmYwKw53BWpOMblo7A==", + "dev": true, + "requires": { + "camel-case": "^4.1.2", + "capital-case": "^1.0.4", + "constant-case": "^3.0.4", + "dot-case": "^3.0.4", + "header-case": "^2.0.4", + "no-case": "^3.0.4", + "param-case": "^3.0.4", + "pascal-case": "^3.1.2", + "path-case": "^3.0.4", + "sentence-case": "^3.0.4", + "snake-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "char-regex": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", + "integrity": "sha512-kWWXztvZ5SBQV+eRgKFeh8q5sLuZY2+8WUIzlxWVTg+oGwY14qylx1KbKzHd8P6ZYkAg0xyIDU9JMHhyJMZ1jw==", + "dev": true + }, + "chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, + "chevrotain": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/chevrotain/-/chevrotain-7.1.1.tgz", + "integrity": "sha512-wy3mC1x4ye+O+QkEinVJkPf5u2vsrDIYW9G7ZuwFl6v/Yu0LwUuT2POsb+NUWApebyxfkQq6+yDfRExbnI5rcw==", + "dev": true, + "requires": { + "regexp-to-ast": "0.5.0" + } + }, + "chownr": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", + "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", + "dev": true + }, + "ci-info": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.3.0.tgz", + "integrity": "sha512-riT/3vI5YpVH6/qomlDnJow6TBee2PBKSEpx3O32EGPYbWGIRsIlGRms3Sm74wYE1JMo8RnO04Hb12+v1J5ICw==", + "dev": true + }, + "cjs-module-lexer": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/cjs-module-lexer/-/cjs-module-lexer-1.2.2.tgz", + "integrity": "sha512-cOU9usZw8/dXIXKtwa8pM0OTJQuJkxMN6w30csNRUerHfeQ5R6U3kkU/FtJeIf3M202OHfY2U8ccInBG7/xogA==", + "dev": true + }, + "class-utils": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", + "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", + "dev": true, + "requires": { + "arr-union": "^3.1.0", + "define-property": "^0.2.5", + "isobject": "^3.0.0", + "static-extend": "^0.1.1" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "clean-git-ref": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/clean-git-ref/-/clean-git-ref-2.0.1.tgz", + "integrity": "sha512-bLSptAy2P0s6hU4PzuIMKmMJJSE6gLXGH1cntDu7bWJUksvuM+7ReOK61mozULErYvP6a15rnYl0zFDef+pyPw==", + "dev": true + }, + "clean-stack": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-3.0.1.tgz", + "integrity": "sha512-lR9wNiMRcVQjSB3a7xXGLuz4cr4wJuuXlaAEbRutGowQTmlp7R72/DOgN21e8jdwblMWl9UOJMJXarX94pzKdg==", + "dev": true, + "requires": { + "escape-string-regexp": "4.0.0" + }, + "dependencies": { + "escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true + } + } + }, + "cli-boxes": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/cli-boxes/-/cli-boxes-1.0.0.tgz", + "integrity": "sha1-T6kXw+WclKAEzWH47lCdplFocUM=", + "dev": true + }, + "cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "requires": { + "restore-cursor": "^3.1.0" + } + }, + "cli-progress": { + "version": "3.10.0", + "resolved": "https://registry.npmjs.org/cli-progress/-/cli-progress-3.10.0.tgz", + "integrity": "sha512-kLORQrhYCAtUPLZxqsAt2YJGOvRdt34+O6jl5cQGb7iF3dM55FQZlTR+rQyIK9JUcO9bBMwZsTlND+3dmFU2Cw==", + "dev": true, + "requires": { + "string-width": "^4.2.0" + } + }, + "cli-table": { + "version": "0.3.11", + "resolved": "https://registry.npmjs.org/cli-table/-/cli-table-0.3.11.tgz", + "integrity": "sha512-IqLQi4lO0nIB4tcdTpN4LCB9FI3uqrJZK7RC515EnhZ6qBaglkIgICb1wjeAqpdoOabm1+SuQtkXIPdYC93jhQ==", + "dev": true, + "requires": { + "colors": "1.0.3" + }, + "dependencies": { + "colors": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.0.3.tgz", + "integrity": "sha1-BDP0TYCWgP3rYO0mDxsMJi6CpAs=", + "dev": true + } + } + }, + "cli-table3": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/cli-table3/-/cli-table3-0.6.2.tgz", + "integrity": "sha512-QyavHCaIC80cMivimWu4aWHilIpiDpfm3hGmqAmXVL1UsnbLuBSMd21hTX6VY4ZSDSM73ESLeF8TOYId3rBTbw==", + "dev": true, + "requires": { + "@colors/colors": "1.5.0", + "string-width": "^4.2.0" + } + }, + "cli-truncate": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-truncate/-/cli-truncate-3.1.0.tgz", + "integrity": "sha512-wfOBkjXteqSnI59oPcJkcPl/ZmwvMMOj340qUIY1SKZCv0B9Cf4D4fAucRkIKQmsIuYK3x1rrgU7MeGRruiuiA==", + "dev": true, + "requires": { + "slice-ansi": "^5.0.0", + "string-width": "^5.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-6.0.1.tgz", + "integrity": "sha512-n5M855fKb2SsfMIiFFoVrABHJC8QtHwVx+mHWP3QcEqBHYienj5dHSgjbxtC0WEZXYt4wcD6zrQElDPhFuZgfA==", + "dev": true + }, + "ansi-styles": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-6.1.0.tgz", + "integrity": "sha512-VbqNsoz55SYGczauuup0MFUyXNQviSpFTj1RQtFzmQLk18qbVSpTFFGMT293rmDaQuKCT6InmbuEyUne4mTuxQ==", + "dev": true + }, + "emoji-regex": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-9.2.2.tgz", + "integrity": "sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg==", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-4.0.0.tgz", + "integrity": "sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ==", + "dev": true + }, + "slice-ansi": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-5.0.0.tgz", + "integrity": "sha512-FC+lgizVPfie0kkhqUScwRu1O/lF6NOgJmlCgK+/LYxDCTk8sGelYaHDhFcDN+Sn3Cv+3VSa4Byeo+IMCzpMgQ==", + "dev": true, + "requires": { + "ansi-styles": "^6.0.0", + "is-fullwidth-code-point": "^4.0.0" + } + }, + "string-width": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-5.1.2.tgz", + "integrity": "sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA==", + "dev": true, + "requires": { + "eastasianwidth": "^0.2.0", + "emoji-regex": "^9.2.2", + "strip-ansi": "^7.0.1" + } + }, + "strip-ansi": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-7.0.1.tgz", + "integrity": "sha512-cXNxvT8dFNRVfhVME3JAe98mkXDYN2O1l7jmcwMnOslDeESg1rF/OZMtK0nRAhiari1unG5cD4jG3rapUAkLbw==", + "dev": true, + "requires": { + "ansi-regex": "^6.0.1" + } + } + } + }, + "cli-ux": { + "version": "4.9.3", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-4.9.3.tgz", + "integrity": "sha512-/1owvF0SZ5Gn54cgrikJ0QskgTzeg30HGjkmjFoaHDJzAqFpuX1DBpFR8aLvsE1J5s9MgeYRENQK4BFwOag5VA==", + "dev": true, + "requires": { + "@oclif/errors": "^1.2.2", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.3", + "ansi-escapes": "^3.1.0", + "ansi-styles": "^3.2.1", + "cardinal": "^2.1.1", + "chalk": "^2.4.1", + "clean-stack": "^2.0.0", + "extract-stack": "^1.0.0", + "fs-extra": "^7.0.0", + "hyperlinker": "^1.0.0", + "indent-string": "^3.2.0", + "is-wsl": "^1.1.0", + "lodash": "^4.17.11", + "password-prompt": "^1.0.7", + "semver": "^5.6.0", + "strip-ansi": "^5.0.0", + "supports-color": "^5.5.0", + "supports-hyperlinks": "^1.0.1", + "treeify": "^1.1.0", + "tslib": "^1.9.3" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true + }, + "clean-stack": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", + "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", + "dev": true + }, + "fs-extra": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", + "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "indent-string": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-3.2.0.tgz", + "integrity": "sha1-Sl/W0nzDMvN+VBmlBNu4NxBckok=", + "dev": true + }, + "is-wsl": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", + "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", + "dev": true + }, + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + } + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + } + } + }, + "cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true + }, + "cliui": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", + "dev": true, + "requires": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^7.0.0" + } + }, + "clone": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", + "integrity": "sha1-G39Ln1kfHo+DZwQBYANFoCiHQ18=", + "dev": true + }, + "clone-buffer": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/clone-buffer/-/clone-buffer-1.0.0.tgz", + "integrity": "sha1-4+JbIHrE5wGvch4staFnksrD3Fg=", + "dev": true + }, + "clone-deep": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", + "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4", + "kind-of": "^6.0.2", + "shallow-clone": "^3.0.0" + } + }, + "clone-response": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/clone-response/-/clone-response-1.0.2.tgz", + "integrity": "sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws=", + "dev": true, + "requires": { + "mimic-response": "^1.0.0" + } + }, + "clone-stats": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/clone-stats/-/clone-stats-1.0.0.tgz", + "integrity": "sha1-s3gt/4u1R04Yuba/D9/ngvh3doA=", + "dev": true + }, + "cloneable-readable": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/cloneable-readable/-/cloneable-readable-1.1.3.tgz", + "integrity": "sha512-2EF8zTQOxYq70Y4XKtorQupqF0m49MBz2/yf5Bj+MHjvpG3Hy7sImifnqD6UA+TKYxeSV+u6qqQPawN5UvnpKQ==", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "process-nextick-args": "^2.0.0", + "readable-stream": "^2.3.5" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } + } + }, + "cls-hooked": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/cls-hooked/-/cls-hooked-4.2.2.tgz", + "integrity": "sha512-J4Xj5f5wq/4jAvcdgoGsL3G103BtWpZrMo8NEinRltN+xpTZdI+M38pyQqhuFU/P792xkMFvnKSf+Lm81U1bxw==", + "dev": true, + "requires": { + "async-hook-jl": "^1.7.6", + "emitter-listener": "^1.0.1", + "semver": "^5.4.1" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + } + } + }, + "co": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/co/-/co-4.6.0.tgz", + "integrity": "sha1-bqa989hTrlTMuOR7+gvz+QMfsYQ=", + "dev": true + }, + "co-prompt": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/co-prompt/-/co-prompt-1.0.0.tgz", + "integrity": "sha1-+zcOntrEhXayenMv5dfyHZ/G5vY=", + "dev": true, + "requires": { + "keypress": "~0.2.1" + } + }, + "code-point-at": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", + "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", + "dev": true + }, + "coffeescript": { + "version": "1.12.7", + "resolved": "https://registry.npmjs.org/coffeescript/-/coffeescript-1.12.7.tgz", + "integrity": "sha512-pLXHFxQMPklVoEekowk8b3erNynC+DVJzChxS/LCBBgR6/8AJkHivkm//zbowcfc7BTCAjryuhx6gPqPRfsFoA==", + "dev": true + }, + "collect-all": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/collect-all/-/collect-all-1.0.4.tgz", + "integrity": "sha512-RKZhRwJtJEP5FWul+gkSMEnaK6H3AGPTTWOiRimCcs+rc/OmQE3Yhy1Q7A7KsdkG3ZXVdZq68Y6ONSdvkeEcKA==", + "dev": true, + "requires": { + "stream-connect": "^1.0.2", + "stream-via": "^1.0.4" + } + }, + "collect-v8-coverage": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/collect-v8-coverage/-/collect-v8-coverage-1.0.1.tgz", + "integrity": "sha512-iBPtljfCNcTKNAto0KEtDfZ3qzjJvqE3aTGZsbhjSBlorqpXJlaWWtPO35D+ZImoC3KWejX64o+yPGxhWSTzfg==", + "dev": true + }, + "collection-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", + "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", + "dev": true, + "requires": { + "map-visit": "^1.0.0", + "object-visit": "^1.0.0" + } + }, + "color-convert": { + "version": "1.9.3", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", + "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", + "dev": true, + "requires": { + "color-name": "1.1.3" + } + }, + "color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", + "dev": true + }, + "colorette": { + "version": "2.0.16", + "resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.16.tgz", + "integrity": "sha512-hUewv7oMjCp+wkBv5Rm0v87eJhq4woh5rSR+42YSQJKecCqgIqNkZ6lAlQms/BwHPJA5NKMRlpxPRv0n8HQW6g==", + "dev": true + }, + "colors": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", + "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==", + "dev": true + }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "requires": { + "delayed-stream": "~1.0.0" + } + }, + "command-line-args": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/command-line-args/-/command-line-args-5.2.1.tgz", + "integrity": "sha512-H4UfQhZyakIjC74I9d34fGYDwk3XpSr17QhEd0Q3I9Xq1CETHo4Hcuo87WyWHpAF1aSLjLRf5lD9ZGX2qStUvg==", + "dev": true, + "requires": { + "array-back": "^3.1.0", + "find-replace": "^3.0.0", + "lodash.camelcase": "^4.3.0", + "typical": "^4.0.0" + }, + "dependencies": { + "array-back": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz", + "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==", + "dev": true + }, + "typical": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/typical/-/typical-4.0.0.tgz", + "integrity": "sha512-VAH4IvQ7BDFYglMd7BPRDfLgxZZX4O4TFcRDA6EN5X7erNJJq+McIEp8np9aVtxrCJ6qx4GTYVfOWNjcqwZgRw==", + "dev": true + } + } + }, + "command-line-tool": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/command-line-tool/-/command-line-tool-0.8.0.tgz", + "integrity": "sha512-Xw18HVx/QzQV3Sc5k1vy3kgtOeGmsKIqwtFFoyjI4bbcpSgnw2CWVULvtakyw4s6fhyAdI6soQQhXc2OzJy62g==", + "dev": true, + "requires": { + "ansi-escape-sequences": "^4.0.0", + "array-back": "^2.0.0", + "command-line-args": "^5.0.0", + "command-line-usage": "^4.1.0", + "typical": "^2.6.1" + }, + "dependencies": { + "array-back": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-2.0.0.tgz", + "integrity": "sha512-eJv4pLLufP3g5kcZry0j6WXpIbzYw9GUB4mVJZno9wfwiBxbizTnHCw3VJb07cBihbFX48Y7oSrW9y+gt4glyw==", + "dev": true, + "requires": { + "typical": "^2.6.1" + } + } + } + }, + "command-line-usage": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/command-line-usage/-/command-line-usage-4.1.0.tgz", + "integrity": "sha512-MxS8Ad995KpdAC0Jopo/ovGIroV/m0KHwzKfXxKag6FHOkGsH8/lv5yjgablcRxCJJC0oJeUMuO/gmaq+Wq46g==", + "dev": true, + "requires": { + "ansi-escape-sequences": "^4.0.0", + "array-back": "^2.0.0", + "table-layout": "^0.4.2", + "typical": "^2.6.1" + }, + "dependencies": { + "array-back": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-2.0.0.tgz", + "integrity": "sha512-eJv4pLLufP3g5kcZry0j6WXpIbzYw9GUB4mVJZno9wfwiBxbizTnHCw3VJb07cBihbFX48Y7oSrW9y+gt4glyw==", + "dev": true, + "requires": { + "typical": "^2.6.1" + } + } + } + }, + "commander": { + "version": "2.20.3", + "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", + "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", + "dev": true + }, + "common-sequence": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/common-sequence/-/common-sequence-2.0.2.tgz", + "integrity": "sha512-jAg09gkdkrDO9EWTdXfv80WWH3yeZl5oT69fGfedBNS9pXUKYInVJ1bJ+/ht2+Moeei48TmSbQDYMc8EOx9G0g==", + "dev": true + }, + "commondir": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", + "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", + "dev": true + }, + "component-emitter": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", + "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", + "dev": true + }, + "compress-brotli": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/compress-brotli/-/compress-brotli-1.3.6.tgz", + "integrity": "sha512-au99/GqZtUtiCBliqLFbWlhnCxn+XSYjwZ77q6mKN4La4qOXDoLVPZ50iXr0WmAyMxl8yqoq3Yq4OeQNPPkyeQ==", + "dev": true, + "requires": { + "@types/json-buffer": "~3.0.0", + "json-buffer": "~3.0.1" + } + }, + "compress-commons": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/compress-commons/-/compress-commons-4.1.1.tgz", + "integrity": "sha512-QLdDLCKNV2dtoTorqgxngQCMA+gWXkM/Nwu7FpeBhk/RdkzimqC3jueb/FDmaZeXh+uby1jkBqE3xArsLBE5wQ==", + "dev": true, + "requires": { + "buffer-crc32": "^0.2.13", + "crc32-stream": "^4.0.2", + "normalize-path": "^3.0.0", + "readable-stream": "^3.6.0" + } + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" + }, + "concurrently": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/concurrently/-/concurrently-7.1.0.tgz", + "integrity": "sha512-Bz0tMlYKZRUDqJlNiF/OImojMB9ruKUz6GCfmhFnSapXgPe+3xzY4byqoKG9tUZ7L2PGEUjfLPOLfIX3labnmw==", + "dev": true, + "requires": { + "chalk": "^4.1.0", + "date-fns": "^2.16.1", + "lodash": "^4.17.21", + "rxjs": "^6.6.3", + "spawn-command": "^0.0.2-1", + "supports-color": "^8.1.0", + "tree-kill": "^1.2.2", + "yargs": "^16.2.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "config-master": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/config-master/-/config-master-3.1.0.tgz", + "integrity": "sha1-ZnZjWQUFooO/JqSE1oSJ10xUhdo=", + "dev": true, + "requires": { + "walk-back": "^2.0.1" + }, + "dependencies": { + "walk-back": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/walk-back/-/walk-back-2.0.1.tgz", + "integrity": "sha1-VU4qnYdPrEeoywBr9EwvDEmYoKQ=", + "dev": true + } + } + }, + "confusing-browser-globals": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/confusing-browser-globals/-/confusing-browser-globals-1.0.10.tgz", + "integrity": "sha512-gNld/3lySHwuhaVluJUKLePYirM3QNCKzVxqAdhJII9/WXKVX5PURzMVJspS1jTslSqjeuG4KMVTSouit5YPHA==", + "dev": true + }, + "constant-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/constant-case/-/constant-case-3.0.4.tgz", + "integrity": "sha512-I2hSBi7Vvs7BEuJDr5dDHfzb/Ruj3FyvFyh7KLilAjNQw3Be+xgqUBA2W6scVEcL0hL1dwPRtIqEPVUCKkSsyQ==", + "dev": true, + "requires": { + "no-case": "^3.0.4", + "tslib": "^2.0.3", + "upper-case": "^2.0.2" + } + }, + "contains-path": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/contains-path/-/contains-path-0.1.0.tgz", + "integrity": "sha1-/ozxhP9mcLa67wGp1IYaXL7EEgo=", + "dev": true + }, + "content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==", + "dev": true + }, + "continuation-local-storage": { + "version": "3.2.1", + "resolved": "https://registry.npmjs.org/continuation-local-storage/-/continuation-local-storage-3.2.1.tgz", + "integrity": "sha512-jx44cconVqkCEEyLSKWwkvUXwO561jXMa3LPjTPsm5QR22PA0/mhe33FT4Xb5y74JDvt/Cq+5lm8S8rskLv9ZA==", + "dev": true, + "requires": { + "async-listener": "^0.6.0", + "emitter-listener": "^1.1.1" + } + }, + "convert-source-map": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.8.0.tgz", + "integrity": "sha512-+OQdjP49zViI/6i7nIJpA8rAl4sV/JdPfU9nZs3VqOwGIgizICvuN2ru6fMd+4llL0tar18UYJXfZ/TWtmhUjA==", + "dev": true, + "requires": { + "safe-buffer": "~5.1.1" + }, + "dependencies": { + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } + } + }, + "copy-descriptor": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", + "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", + "dev": true + }, + "core-js": { + "version": "3.21.1", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.21.1.tgz", + "integrity": "sha512-FRq5b/VMrWlrmCzwRrpDYNxyHP9BcAZC+xHJaqTgIE5091ZV1NTmyh0sGOg5XqpnHvR0svdy0sv1gWA1zmhxig==", + "dev": true + }, + "core-js-pure": { + "version": "3.21.1", + "resolved": "https://registry.npmjs.org/core-js-pure/-/core-js-pure-3.21.1.tgz", + "integrity": "sha512-12VZfFIu+wyVbBebyHmRTuEE/tZrB4tJToWcwAMcsp3h4+sHR+fMJWbKpYiCRWlhFBq+KNyO8rIV9rTkeVmznQ==", + "dev": true + }, + "core-util-is": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", + "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", + "dev": true + }, + "crc-32": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/crc-32/-/crc-32-1.2.2.tgz", + "integrity": "sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==", + "dev": true + }, + "crc32-stream": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/crc32-stream/-/crc32-stream-4.0.2.tgz", + "integrity": "sha512-DxFZ/Hk473b/muq1VJ///PMNLj0ZMnzye9thBpmjpJKCc5eMgB95aK8zCGrGfQ90cWo561Te6HK9D+j4KPdM6w==", + "dev": true, + "requires": { + "crc-32": "^1.2.0", + "readable-stream": "^3.4.0" + } + }, + "create-error-class": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/create-error-class/-/create-error-class-3.0.2.tgz", + "integrity": "sha1-Br56vvlHo/FKMP1hBnHUAbyot7Y=", + "dev": true, + "requires": { + "capture-stack-trace": "^1.0.0" + } + }, + "cross-spawn": { + "version": "6.0.5", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", + "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", + "dev": true, + "requires": { + "nice-try": "^1.0.4", + "path-key": "^2.0.1", + "semver": "^5.5.0", + "shebang-command": "^1.2.0", + "which": "^1.2.9" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + } + } + }, + "csprng": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/csprng/-/csprng-0.1.2.tgz", + "integrity": "sha1-S8aPEvo2jSUqWYQcusqXSxirReI=", + "dev": true, + "requires": { + "sequin": "*" + } + }, + "cssesc": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", + "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", + "dev": true + }, + "cssom": { + "version": "0.4.4", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.4.4.tgz", + "integrity": "sha512-p3pvU7r1MyyqbTk+WbNJIgJjG2VmTIaB10rI93LzVPrmDJKkzKYMtxxyAvQXR/NS6otuzveI7+7BBq3SjBS2mw==", + "dev": true + }, + "cssstyle": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-2.3.0.tgz", + "integrity": "sha512-AZL67abkUzIuvcHqk7c09cezpGNcxUxU4Ioi/05xHk4DQeTkWmGYftIE6ctU6AEt+Gn4n1lDStOtj7FKycP71A==", + "dev": true, + "requires": { + "cssom": "~0.3.6" + }, + "dependencies": { + "cssom": { + "version": "0.3.8", + "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", + "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==", + "dev": true + } + } + }, + "csv-parse": { + "version": "4.16.3", + "resolved": "https://registry.npmjs.org/csv-parse/-/csv-parse-4.16.3.tgz", + "integrity": "sha512-cO1I/zmz4w2dcKHVvpCr7JVRu8/FymG5OEpmvsZYlccYolPBLoVGKUHgNoc4ZGkFeFlWGEDmMyBM+TTqRdW/wg==", + "dev": true + }, + "csv-stringify": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/csv-stringify/-/csv-stringify-1.1.2.tgz", + "integrity": "sha1-d6QVJlgbzjOA8SsA18W7rHDIK1g=", + "dev": true, + "requires": { + "lodash.get": "~4.4.2" + } + }, + "dargs": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/dargs/-/dargs-6.1.0.tgz", + "integrity": "sha512-5dVBvpBLBnPwSsYXqfybFyehMmC/EenKEcf23AhCTgTf48JFBbmJKqoZBsERDnjL0FyiVTYWdFsRfTLHxLyKdQ==", + "dev": true + }, + "dashdash": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", + "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", + "dev": true, + "requires": { + "assert-plus": "^1.0.0" + } + }, + "data-uri-to-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/data-uri-to-buffer/-/data-uri-to-buffer-3.0.1.tgz", + "integrity": "sha512-WboRycPNsVw3B3TL559F7kuBUM4d8CgMEvk6xEJlOp7OBPjt6G7z8WMWlD2rOFZLk6OYfFIUGsCOWzcQH9K2og==", + "dev": true + }, + "data-urls": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", + "integrity": "sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ==", + "dev": true, + "requires": { + "abab": "^2.0.3", + "whatwg-mimetype": "^2.3.0", + "whatwg-url": "^8.0.0" + } + }, + "date-fns": { + "version": "2.28.0", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.28.0.tgz", + "integrity": "sha512-8d35hViGYx/QH0icHYCeLmsLmMUheMmTyV9Fcm6gvNwdw31yXXH+O85sOBJ+OLnLQMKZowvpKb6FgMIQjcpvQw==", + "dev": true + }, + "dateformat": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-3.0.3.tgz", + "integrity": "sha512-jyCETtSl3VMZMWeRo7iY1FL19ges1t55hMo5yaam4Jrsm5EPL89UQkoQRyiI+Yf4k8r2ZpdngkV8hr1lIdjb3Q==", + "dev": true + }, + "dayjs": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/dayjs/-/dayjs-1.11.0.tgz", + "integrity": "sha512-JLC809s6Y948/FuCZPm5IX8rRhQwOiyMb2TfVVQEixG7P8Lm/gt5S7yoQZmC8x1UehI9Pb7sksEt4xx14m+7Ug==", + "dev": true + }, + "dayjs-plugin-utc": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/dayjs-plugin-utc/-/dayjs-plugin-utc-0.1.2.tgz", + "integrity": "sha512-ExERH5o3oo6jFOdkvMP3gytTCQ9Ksi5PtylclJWghr7k7m3o2U5QrwtdiJkOxLOH4ghr0EKhpqGefzGz1VvVJg==", + "dev": true + }, + "debug": { + "version": "4.3.4", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", + "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "dev": true, + "requires": { + "ms": "2.1.2" + } + }, + "debuglog": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/debuglog/-/debuglog-1.0.1.tgz", + "integrity": "sha1-qiT/uaw9+aI1GDfPstJ5NgzXhJI=", + "dev": true + }, + "decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", + "dev": true + }, + "decimal.js": { + "version": "10.3.1", + "resolved": "https://registry.npmjs.org/decimal.js/-/decimal.js-10.3.1.tgz", + "integrity": "sha512-V0pfhfr8suzyPGOx3nmq4aHqabehUZn6Ch9kyFpV79TGDTWFmHqUqXdabR7QHqxzrYolF4+tVmJhUG4OURg5dQ==", + "dev": true + }, + "decode-uri-component": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", + "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", + "dev": true + }, + "decompress-response": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz", + "integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==", + "requires": { + "mimic-response": "^3.1.0" + }, + "dependencies": { + "mimic-response": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz", + "integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==" + } + } + }, + "dedent": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/dedent/-/dedent-0.7.0.tgz", + "integrity": "sha1-JJXduvbrh0q7Dhvp3yLS5aVEMmw=", + "dev": true + }, + "deep-extend": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz", + "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==", + "dev": true + }, + "deep-is": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", + "integrity": "sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ==", + "dev": true + }, + "deepmerge": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/deepmerge/-/deepmerge-4.2.2.tgz", + "integrity": "sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg==", + "dev": true + }, + "defer-to-connect": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/defer-to-connect/-/defer-to-connect-2.0.1.tgz", + "integrity": "sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg==", + "dev": true + }, + "define-lazy-prop": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/define-lazy-prop/-/define-lazy-prop-2.0.0.tgz", + "integrity": "sha512-Ds09qNh8yw3khSjiJjiUInaGX9xlqZDY7JVryGxdxV7NPeuqQfplOpQ66yJFZut3jLa5zOwkXw1g9EI2uKh4Og==", + "dev": true + }, + "define-properties": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", + "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", + "dev": true, + "requires": { + "object-keys": "^1.0.12" + } + }, + "define-property": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", + "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", + "dev": true, + "requires": { + "is-descriptor": "^1.0.2", + "isobject": "^3.0.1" + }, + "dependencies": { + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "degenerator": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/degenerator/-/degenerator-3.0.2.tgz", + "integrity": "sha512-c0mef3SNQo56t6urUU6tdQAs+ThoD0o9B9MJ8HEt7NQcGEILCRFqQb7ZbP9JAv+QF1Ky5plydhMR/IrqWDm+TQ==", + "dev": true, + "requires": { + "ast-types": "^0.13.2", + "escodegen": "^1.8.1", + "esprima": "^4.0.0", + "vm2": "^3.9.8" + }, + "dependencies": { + "escodegen": { + "version": "1.14.3", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.3.tgz", + "integrity": "sha512-qFcX0XJkdg+PB3xjZZG/wKSuT1PnQWx57+TVSjIMmILd2yC/6ByYElPwJnslDsuWuSAp4AwJGumarAAmJch5Kw==", + "dev": true, + "requires": { + "esprima": "^4.0.1", + "estraverse": "^4.2.0", + "esutils": "^2.0.2", + "optionator": "^0.8.1", + "source-map": "~0.6.1" + } + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "optional": true + } + } + }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true + }, + "depd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", + "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==", + "dev": true + }, + "detect-newline": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", + "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", + "dev": true + }, + "dezalgo": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.4.tgz", + "integrity": "sha512-rXSP0bf+5n0Qonsb+SVVfNfIsimO4HEtmnIpPHY8Q1UCzKlQrDMfdobr8nJOOsRgWCyMRqeSBQzmWUMq7zvVig==", + "dev": true, + "requires": { + "asap": "^2.0.0", + "wrappy": "1" + } + }, + "diagnostic-channel": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/diagnostic-channel/-/diagnostic-channel-0.3.1.tgz", + "integrity": "sha512-6eb9YRrimz8oTr5+JDzGmSYnXy5V7YnK5y/hd8AUDK1MssHjQKm9LlD6NSrHx4vMDF3+e/spI2hmWTviElgWZA==", + "dev": true, + "requires": { + "semver": "^5.3.0" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + } + } + }, + "diagnostic-channel-publishers": { + "version": "0.4.4", + "resolved": "https://registry.npmjs.org/diagnostic-channel-publishers/-/diagnostic-channel-publishers-0.4.4.tgz", + "integrity": "sha512-l126t01d2ZS9EreskvEtZPrcgstuvH3rbKy82oUhUrVmBaGx4hO9wECdl3cvZbKDYjMF3QJDB5z5dL9yWAjvZQ==", + "dev": true + }, + "diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true + }, + "diff-sequences": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/diff-sequences/-/diff-sequences-27.5.1.tgz", + "integrity": "sha512-k1gCAXAsNgLwEL+Y8Wvl+M6oEFj5bgazfZULpS5CneoPPXRaCCW7dm+q21Ky2VEE5X+VeRDBVg1Pcvvsr4TtNQ==", + "dev": true + }, + "diff3": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/diff3/-/diff3-0.0.3.tgz", + "integrity": "sha1-1OXDpM305f4SEatC5pP8tDIVgPw=", + "dev": true + }, + "dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "requires": { + "path-type": "^4.0.0" + } + }, + "dmd": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/dmd/-/dmd-6.1.0.tgz", + "integrity": "sha512-0zQIJ873gay1scCTFZvHPWM9mVJBnaylB2NQDI8O9u8O32m00Jb6uxDKexZm8hjTRM7RiWe0FJ32pExHoXdwoQ==", + "dev": true, + "requires": { + "array-back": "^6.2.2", + "cache-point": "^2.0.0", + "common-sequence": "^2.0.2", + "file-set": "^4.0.2", + "handlebars": "^4.7.7", + "marked": "^4.0.12", + "object-get": "^2.1.1", + "reduce-flatten": "^3.0.1", + "reduce-unique": "^2.0.1", + "reduce-without": "^1.0.1", + "test-value": "^3.0.0", + "walk-back": "^5.1.0" + }, + "dependencies": { + "marked": { + "version": "4.0.14", + "resolved": "https://registry.npmjs.org/marked/-/marked-4.0.14.tgz", + "integrity": "sha512-HL5sSPE/LP6U9qKgngIIPTthuxC0jrfxpYMZ3LdGDD3vTnLs59m2Z7r6+LNDR3ToqEQdkKd6YaaEfJhodJmijQ==", + "dev": true + }, + "reduce-flatten": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/reduce-flatten/-/reduce-flatten-3.0.1.tgz", + "integrity": "sha512-bYo+97BmUUOzg09XwfkwALt4PQH1M5L0wzKerBt6WLm3Fhdd43mMS89HiT1B9pJIqko/6lWx3OnV4J9f2Kqp5Q==", + "dev": true + } + } + }, + "doctrine": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-1.5.0.tgz", + "integrity": "sha1-N53Ocw9hZvds76TmcHoVmwLFpvo=", + "dev": true, + "requires": { + "esutils": "^2.0.2", + "isarray": "^1.0.0" + } + }, + "domexception": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/domexception/-/domexception-2.0.1.tgz", + "integrity": "sha512-yxJ2mFy/sibVQlu5qHjOkf9J3K6zgmCxgJ94u2EdvDOV09H+32LtRswEcUsmUWN72pVLOEnTSRaIVVzVQgS0dg==", + "dev": true, + "requires": { + "webidl-conversions": "^5.0.0" + }, + "dependencies": { + "webidl-conversions": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-5.0.0.tgz", + "integrity": "sha512-VlZwKPCkYKxQgeSbH5EyngOmRp7Ww7I9rQLERETtf5ofd9pGeswWiOtogpEO850jziPRarreGxn5QIiTqpb2wA==", + "dev": true + } + } + }, + "dot-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/dot-case/-/dot-case-3.0.4.tgz", + "integrity": "sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==", + "dev": true, + "requires": { + "no-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "download-stats": { + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/download-stats/-/download-stats-0.3.4.tgz", + "integrity": "sha512-ic2BigbyUWx7/CBbsfGjf71zUNZB4edBGC3oRliSzsoNmvyVx3Ycfp1w3vp2Y78Ee0eIIkjIEO5KzW0zThDGaA==", + "dev": true, + "requires": { + "JSONStream": "^1.2.1", + "lazy-cache": "^2.0.1", + "moment": "^2.15.1" + } + }, + "dtrace-provider": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/dtrace-provider/-/dtrace-provider-0.6.0.tgz", + "integrity": "sha1-CweNVReTfYcxAUUtkUZzdVe3XlE=", + "dev": true, + "optional": true, + "requires": { + "nan": "^2.0.8" + } + }, + "duplexer2": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.1.4.tgz", + "integrity": "sha1-ixLauHjA1p4+eJEFFmKjL8a93ME=", + "dev": true, + "requires": { + "readable-stream": "^2.0.2" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } + } + }, + "duplexer3": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/duplexer3/-/duplexer3-0.1.4.tgz", + "integrity": "sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=", + "dev": true + }, + "eastasianwidth": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", + "integrity": "sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA==", + "dev": true + }, + "ecc-jsbn": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", + "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", + "dev": true, + "requires": { + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" + } + }, + "ecdsa-sig-formatter": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz", + "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==", + "dev": true, + "requires": { + "safe-buffer": "^5.0.1" + } + }, + "editions": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/editions/-/editions-2.3.1.tgz", + "integrity": "sha512-ptGvkwTvGdGfC0hfhKg0MT+TRLRKGtUiWGBInxOm5pz7ssADezahjCUaYuZ8Dr+C05FW0AECIIPt4WBxVINEhA==", + "dev": true, + "requires": { + "errlop": "^2.0.0", + "semver": "^6.3.0" + } + }, + "ejs": { + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.6.tgz", + "integrity": "sha512-9lt9Zse4hPucPkoP7FHDF0LQAlGyF9JVpnClFLFH3aSSbxmyoqINRpp/9wePWJTUl4KOQwRL72Iw3InHPDkoGw==", + "dev": true, + "requires": { + "jake": "^10.6.1" + } + }, + "electron-to-chromium": { + "version": "1.4.107", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.107.tgz", + "integrity": "sha512-Huen6taaVrUrSy8o7mGStByba8PfOWWluHNxSHGBrCgEdFVLtvdQDBr9LBCF9Uci8SYxh28QNNMO0oC17wbGAg==", + "dev": true + }, + "emitter-listener": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/emitter-listener/-/emitter-listener-1.1.2.tgz", + "integrity": "sha512-Bt1sBAGFHY9DKY+4/2cV6izcKJUf5T7/gkdmkxzX/qv9CcGH8xSwVRW5mtX03SWJtRTWSOpzCuWN9rBFYZepZQ==", + "dev": true, + "requires": { + "shimmer": "^1.2.0" + } + }, + "emittery": { + "version": "0.8.1", + "resolved": "https://registry.npmjs.org/emittery/-/emittery-0.8.1.tgz", + "integrity": "sha512-uDfvUjVrfGJJhymx/kz6prltenw1u7WrCg1oa94zYY8xxVpLLUu045LAT0dhDZdXG58/EpPL/5kA180fQ/qudg==", + "dev": true + }, + "emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "end-of-stream": { + "version": "1.4.4", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", + "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", + "dev": true, + "requires": { + "once": "^1.4.0" + } + }, + "errlop": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/errlop/-/errlop-2.2.0.tgz", + "integrity": "sha512-e64Qj9+4aZzjzzFpZC7p5kmm/ccCrbLhAJplhsDXQFs87XTsXwOpH4s1Io2s90Tau/8r2j9f4l/thhDevRjzxw==", + "dev": true + }, + "error": { + "version": "7.2.1", + "resolved": "https://registry.npmjs.org/error/-/error-7.2.1.tgz", + "integrity": "sha512-fo9HBvWnx3NGUKMvMwB/CBCMMrfEJgbDTVDEkPygA3Bdd3lM1OyCd+rbQ8BwnpF6GdVeOLDNmyL4N5Bg80ZvdA==", + "dev": true, + "requires": { + "string-template": "~0.2.1" + } + }, + "error-ex": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", + "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", + "dev": true, + "requires": { + "is-arrayish": "^0.2.1" + } + }, + "es-abstract": { + "version": "1.19.4", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.19.4.tgz", + "integrity": "sha512-flV8e5g9/xulChMG48Fygk1ptpo4lQRJ0eJYtxJFgi7pklLx7EFcOJ34jnvr8pbWlaFN/AT1cZpe0hiFel9Hqg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "es-to-primitive": "^1.2.1", + "function-bind": "^1.1.1", + "get-intrinsic": "^1.1.1", + "get-symbol-description": "^1.0.0", + "has": "^1.0.3", + "has-symbols": "^1.0.3", + "internal-slot": "^1.0.3", + "is-callable": "^1.2.4", + "is-negative-zero": "^2.0.2", + "is-regex": "^1.1.4", + "is-shared-array-buffer": "^1.0.2", + "is-string": "^1.0.7", + "is-weakref": "^1.0.2", + "object-inspect": "^1.12.0", + "object-keys": "^1.1.1", + "object.assign": "^4.1.2", + "string.prototype.trimend": "^1.0.4", + "string.prototype.trimstart": "^1.0.4", + "unbox-primitive": "^1.0.1" + } + }, + "es-shim-unscopables": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-shim-unscopables/-/es-shim-unscopables-1.0.0.tgz", + "integrity": "sha512-Jm6GPcCdC30eMLbZ2x8z2WuRwAws3zTBBKuusffYVUrNj/GVSUAZ+xKMaUpfNDR5IbyNA5LJbaecoUVbmUcB1w==", + "dev": true, + "requires": { + "has": "^1.0.3" + } + }, + "es-to-primitive": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", + "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", + "dev": true, + "requires": { + "is-callable": "^1.1.4", + "is-date-object": "^1.0.1", + "is-symbol": "^1.0.2" + } + }, + "escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "dev": true + }, + "escape-string-regexp": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", + "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", + "dev": true + }, + "escodegen": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-2.0.0.tgz", + "integrity": "sha512-mmHKys/C8BFUGI+MAWNcSYoORYLMdPzjrknd2Vc+bUsjN5bXcr8EhrNB+UTqfL1y3I9c4fw2ihgtMPQLBRiQxw==", + "dev": true, + "requires": { + "esprima": "^4.0.1", + "estraverse": "^5.2.0", + "esutils": "^2.0.2", + "optionator": "^0.8.1", + "source-map": "~0.6.1" + }, + "dependencies": { + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "optional": true + } + } + }, + "eslint": { + "version": "8.13.0", + "resolved": "https://registry.npmjs.org/eslint/-/eslint-8.13.0.tgz", + "integrity": "sha512-D+Xei61eInqauAyTJ6C0q6x9mx7kTUC1KZ0m0LSEexR0V+e94K12LmWX076ZIsldwfQ2RONdaJe0re0TRGQbRQ==", + "dev": true, + "requires": { + "@eslint/eslintrc": "^1.2.1", + "@humanwhocodes/config-array": "^0.9.2", + "ajv": "^6.10.0", + "chalk": "^4.0.0", + "cross-spawn": "^7.0.2", + "debug": "^4.3.2", + "doctrine": "^3.0.0", + "escape-string-regexp": "^4.0.0", + "eslint-scope": "^7.1.1", + "eslint-utils": "^3.0.0", + "eslint-visitor-keys": "^3.3.0", + "espree": "^9.3.1", + "esquery": "^1.4.0", + "esutils": "^2.0.2", + "fast-deep-equal": "^3.1.3", + "file-entry-cache": "^6.0.1", + "functional-red-black-tree": "^1.0.1", + "glob-parent": "^6.0.1", + "globals": "^13.6.0", + "ignore": "^5.2.0", + "import-fresh": "^3.0.0", + "imurmurhash": "^0.1.4", + "is-glob": "^4.0.0", + "js-yaml": "^4.1.0", + "json-stable-stringify-without-jsonify": "^1.0.1", + "levn": "^0.4.1", + "lodash.merge": "^4.6.2", + "minimatch": "^3.0.4", + "natural-compare": "^1.4.0", + "optionator": "^0.9.1", + "regexpp": "^3.2.0", + "strip-ansi": "^6.0.1", + "strip-json-comments": "^3.1.0", + "text-table": "^0.2.0", + "v8-compile-cache": "^2.0.3" + }, + "dependencies": { + "acorn": { + "version": "8.7.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.7.0.tgz", + "integrity": "sha512-V/LGr1APy+PXIwKebEWrkZPwoeoF+w1jiOBUmuxuiUIaOHtob8Qc9BTrYo7VuI5fR8tqsy+buA2WFooR5olqvQ==", + "dev": true + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "doctrine": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", + "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", + "dev": true + }, + "eslint-scope": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-7.1.1.tgz", + "integrity": "sha512-QKQM/UXpIiHcLqJ5AOyIW7XZmzjkzQXYE54n1++wb0u9V/abW3l9uQnxX8Z5Xd18xyKIMTUAyQ0k1e8pz6LUrw==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^5.2.0" + } + }, + "eslint-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-3.0.0.tgz", + "integrity": "sha512-uuQC43IGctw68pJA1RgbQS8/NP7rch6Cwd4j3ZBtgo4/8Flj4eGE7ZYSZRN3iq5pVUv6GPdW5Z1RFleo84uLDA==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^2.0.0" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", + "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", + "dev": true + } + } + }, + "eslint-visitor-keys": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-3.3.0.tgz", + "integrity": "sha512-mQ+suqKJVyeuwGYHAdjMFqjCyfl8+Ldnxuyp3ldiMBFKkvytrXUZWaiPCEav8qDHKty44bD+qV1IP4T+w+xXRA==", + "dev": true + }, + "espree": { + "version": "9.3.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-9.3.1.tgz", + "integrity": "sha512-bvdyLmJMfwkV3NCRl5ZhJf22zBFo1y8bYh3VYb+bfzqNB4Je68P2sSuXyuFquzWLebHpNd2/d5uv7yoP9ISnGQ==", + "dev": true, + "requires": { + "acorn": "^8.7.0", + "acorn-jsx": "^5.3.1", + "eslint-visitor-keys": "^3.3.0" + } + }, + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + }, + "file-entry-cache": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-6.0.1.tgz", + "integrity": "sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg==", + "dev": true, + "requires": { + "flat-cache": "^3.0.4" + } + }, + "flat-cache": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.0.4.tgz", + "integrity": "sha512-dm9s5Pw7Jc0GvMYbshN6zchCA9RgQlzzEZX3vylR9IqFfS8XciblUXOKfW6SiuJ0e13eDYZoZV5wdrev7P3Nwg==", + "dev": true, + "requires": { + "flatted": "^3.1.0", + "rimraf": "^3.0.2" + } + }, + "flatted": { + "version": "3.2.5", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-3.2.5.tgz", + "integrity": "sha512-WIWGi2L3DyTUvUrwRKgGi9TwxQMUEqPOPQBVi71R96jZXJdFskXEmf54BoZaS1kknGODoIGASGEzBUYdyMCBJg==", + "dev": true + }, + "glob-parent": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", + "dev": true, + "requires": { + "is-glob": "^4.0.3" + } + }, + "globals": { + "version": "13.13.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-13.13.0.tgz", + "integrity": "sha512-EQ7Q18AJlPwp3vUDL4mKA0KXrXyNIQyWon6T6XQiBQF0XHvRsiCSrWmmeATpUzdJN2HhWZU6Pdl0a9zdep5p6A==", + "dev": true, + "requires": { + "type-fest": "^0.20.2" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "js-yaml": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", + "dev": true, + "requires": { + "argparse": "^2.0.1" + } + }, + "levn": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.4.1.tgz", + "integrity": "sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1", + "type-check": "~0.4.0" + } + }, + "optionator": { + "version": "0.9.1", + "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.1.tgz", + "integrity": "sha512-74RlY5FCnhq4jRxVUPKDaRwrVNXMqsGsiW6AJw4XK8hmtm10wC0ypZBLw5IIp85NZMr91+qd1RvvENwg7jjRFw==", + "dev": true, + "requires": { + "deep-is": "^0.1.3", + "fast-levenshtein": "^2.0.6", + "levn": "^0.4.1", + "prelude-ls": "^1.2.1", + "type-check": "^0.4.0", + "word-wrap": "^1.2.3" + } + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "prelude-ls": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", + "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==", + "dev": true + }, + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "type-check": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.4.0.tgz", + "integrity": "sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==", + "dev": true, + "requires": { + "prelude-ls": "^1.2.1" + } + }, + "type-fest": { + "version": "0.20.2", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", + "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", + "dev": true + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, + "eslint-config-prettier": { + "version": "8.5.0", + "resolved": "https://registry.npmjs.org/eslint-config-prettier/-/eslint-config-prettier-8.5.0.tgz", + "integrity": "sha512-obmWKLUNCnhtQRKc+tmnYuQl0pFU1ibYJQ5BGhTVB08bHe9wC8qUeG7c08dj9XX+AuPj1YSGSQIHl1pnDHZR0Q==", + "dev": true + }, + "eslint-config-xo": { + "version": "0.36.0", + "resolved": "https://registry.npmjs.org/eslint-config-xo/-/eslint-config-xo-0.36.0.tgz", + "integrity": "sha512-RCaqCyI38awe3qgiO0Z8CqHs9yw7dMKdV6ZRTFSR7lm0//370tbDEZaQBXnztgpwe5m6D+VvFWc3vLMP/W6EAg==", + "dev": true, + "requires": { + "confusing-browser-globals": "1.0.10" + } + }, + "eslint-config-xo-space": { + "version": "0.27.0", + "resolved": "https://registry.npmjs.org/eslint-config-xo-space/-/eslint-config-xo-space-0.27.0.tgz", + "integrity": "sha512-b8UjW+nQyOkhiANVpIptqlKPyE7XRyQ40uQ1NoBhzVfu95gxfZGrpliq8ZHBpaOF2wCLZaexTSjg7Rvm99vj4A==", + "dev": true, + "requires": { + "eslint-config-xo": "^0.35.0" + }, + "dependencies": { + "eslint-config-xo": { + "version": "0.35.0", + "resolved": "https://registry.npmjs.org/eslint-config-xo/-/eslint-config-xo-0.35.0.tgz", + "integrity": "sha512-+WyZTLWUJlvExFrBU/Ldw8AB/S0d3x+26JQdBWbcqig2ZaWh0zinYcHok+ET4IoPaEcRRf3FE9kjItNVjBwnAg==", + "dev": true, + "requires": { + "confusing-browser-globals": "1.0.10" + } + } + } + }, + "eslint-import-resolver-node": { + "version": "0.3.6", + "resolved": "https://registry.npmjs.org/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.6.tgz", + "integrity": "sha512-0En0w03NRVMn9Uiyn8YRPDKvWjxCWkslUEhGNTdGx15RvPJYQ+lbOlqrlNI2vEAs4pDYK4f/HN2TbDmk5TP0iw==", + "dev": true, + "requires": { + "debug": "^3.2.7", + "resolve": "^1.20.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + } + } + }, + "eslint-module-utils": { + "version": "2.7.3", + "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.7.3.tgz", + "integrity": "sha512-088JEC7O3lDZM9xGe0RerkOMd0EjFl+Yvd1jPWIkMT5u3H9+HC34mWWPnqPrN13gieT9pBOO+Qt07Nb/6TresQ==", + "dev": true, + "requires": { + "debug": "^3.2.7", + "find-up": "^2.1.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "find-up": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", + "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", + "dev": true, + "requires": { + "locate-path": "^2.0.0" + } + }, + "locate-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", + "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", + "dev": true, + "requires": { + "p-locate": "^2.0.0", + "path-exists": "^3.0.0" + } + }, + "p-limit": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", + "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", + "dev": true, + "requires": { + "p-try": "^1.0.0" + } + }, + "p-locate": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", + "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", + "dev": true, + "requires": { + "p-limit": "^1.1.0" + } + }, + "p-try": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", + "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", + "dev": true + }, + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "dev": true + } + } + }, + "eslint-plugin-compat": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-compat/-/eslint-plugin-compat-4.0.2.tgz", + "integrity": "sha512-xqvoO54CLTVaEYGMzhu35Wzwk/As7rCvz/2dqwnFiWi0OJccEtGIn+5qq3zqIu9nboXlpdBN579fZcItC73Ycg==", + "dev": true, + "requires": { + "@mdn/browser-compat-data": "^4.1.5", + "ast-metadata-inferer": "^0.7.0", + "browserslist": "^4.16.8", + "caniuse-lite": "^1.0.30001304", + "core-js": "^3.16.2", + "find-up": "^5.0.0", + "lodash.memoize": "4.1.2", + "semver": "7.3.5" + }, + "dependencies": { + "semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } + } + }, + "eslint-plugin-import": { + "version": "2.26.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.26.0.tgz", + "integrity": "sha512-hYfi3FXaM8WPLf4S1cikh/r4IxnO6zrhZbEGz2b660EJRbuxgpDS5gkCuYgGWg2xxh2rBuIr4Pvhve/7c31koA==", + "dev": true, + "requires": { + "array-includes": "^3.1.4", + "array.prototype.flat": "^1.2.5", + "debug": "^2.6.9", + "doctrine": "^2.1.0", + "eslint-import-resolver-node": "^0.3.6", + "eslint-module-utils": "^2.7.3", + "has": "^1.0.3", + "is-core-module": "^2.8.1", + "is-glob": "^4.0.3", + "minimatch": "^3.1.2", + "object.values": "^1.1.5", + "resolve": "^1.22.0", + "tsconfig-paths": "^3.14.1" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "doctrine": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-2.1.0.tgz", + "integrity": "sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw==", + "dev": true, + "requires": { + "esutils": "^2.0.2" + } + }, + "is-core-module": { + "version": "2.8.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.8.1.tgz", + "integrity": "sha512-SdNCUs284hr40hFTFP6l0IfZ/RSrMXF3qgoRHd3/79unUTvrFO/JoXwkGm+5J/Oe3E/b5GsnG330uUNgRpu1PA==", + "dev": true, + "requires": { + "has": "^1.0.3" + } + }, + "minimatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "resolve": { + "version": "1.22.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.0.tgz", + "integrity": "sha512-Hhtrw0nLeSrFQ7phPp4OOcVjLPIeMnRlr5mcnVuMe7M/7eBn98A3hmFRLoFo3DLZkivSYwhRUJTyPyWAk56WLw==", + "dev": true, + "requires": { + "is-core-module": "^2.8.1", + "path-parse": "^1.0.7", + "supports-preserve-symlinks-flag": "^1.0.0" + } + } + } + }, + "eslint-plugin-jest": { + "version": "23.8.2", + "resolved": "https://registry.npmjs.org/eslint-plugin-jest/-/eslint-plugin-jest-23.8.2.tgz", + "integrity": "sha512-xwbnvOsotSV27MtAe7s8uGWOori0nUsrXh2f1EnpmXua8sDfY6VZhHAhHg2sqK7HBNycRQExF074XSZ7DvfoFg==", + "dev": true, + "requires": { + "@typescript-eslint/experimental-utils": "^2.5.0" + } + }, + "eslint-restricted-globals": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/eslint-restricted-globals/-/eslint-restricted-globals-0.2.0.tgz", + "integrity": "sha512-kwYJALm5KS2QW3Mc1PgObO4V+pTR6RQtRT65L1GQILlEnAhabUQqGAX7/qUjoQR4KZJKehWpBtyDEiDecwmY9A==", + "dev": true + }, + "eslint-scope": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-5.1.1.tgz", + "integrity": "sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==", + "dev": true, + "requires": { + "esrecurse": "^4.3.0", + "estraverse": "^4.1.1" + } + }, + "eslint-utils": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-utils/-/eslint-utils-2.1.0.tgz", + "integrity": "sha512-w94dQYoauyvlDc43XnGB8lU3Zt713vNChgt4EWwhXAP2XkBvndfxF0AgIqKOOasjPIPzj9JqgwkwbCYD0/V3Zg==", + "dev": true, + "requires": { + "eslint-visitor-keys": "^1.1.0" + }, + "dependencies": { + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + } + } + }, + "eslint-visitor-keys": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-2.1.0.tgz", + "integrity": "sha512-0rSmRBzXgDzIsD6mGdJgevzgezI534Cer5L/vyMX0kHzT/jiB43jRhd9YUlMGYLQy2zprNmoT8qasCGtY+QaKw==", + "dev": true + }, + "espree": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/espree/-/espree-6.2.1.tgz", + "integrity": "sha512-ysCxRQY3WaXJz9tdbWOwuWr5Y/XrPTGX9Kiz3yoUXwW0VZ4w30HTkQLaGx/+ttFjF8i+ACbArnB4ce68a9m5hw==", + "dev": true, + "requires": { + "acorn": "^7.1.1", + "acorn-jsx": "^5.2.0", + "eslint-visitor-keys": "^1.1.0" + }, + "dependencies": { + "acorn": { + "version": "7.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.4.1.tgz", + "integrity": "sha512-nQyp0o1/mNdbTO1PO6kHkwSrmgZ0MT/jCCpNiwbUjGoRN4dlBhqJtoQuCnEOKzgTVwg0ZWiCoQy6SxMebQVh8A==", + "dev": true + }, + "eslint-visitor-keys": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/eslint-visitor-keys/-/eslint-visitor-keys-1.3.0.tgz", + "integrity": "sha512-6J72N8UNa462wa/KFODt/PJ3IU60SDpC3QXC1Hjc1BXXpfL2C9R5+AU7jhe0F6GREqVMh4Juu+NY7xn+6dipUQ==", + "dev": true + } + } + }, + "esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true + }, + "esquery": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.4.0.tgz", + "integrity": "sha512-cCDispWt5vHHtwMY2YrAQ4ibFkAL8RbH5YGBnZBc90MolvvfkkQcJro/aZiAQUlQ3qgrYS6D6v8Gc5G5CQsc9w==", + "dev": true, + "requires": { + "estraverse": "^5.1.0" + }, + "dependencies": { + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + } + } + }, + "esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "requires": { + "estraverse": "^5.2.0" + }, + "dependencies": { + "estraverse": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.3.0.tgz", + "integrity": "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==", + "dev": true + } + } + }, + "estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true + }, + "estree-walker": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz", + "integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w==", + "dev": true + }, + "esutils": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", + "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", + "dev": true + }, + "event-target-shim": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/event-target-shim/-/event-target-shim-5.0.1.tgz", + "integrity": "sha512-i/2XbnSz/uxRCU6+NdVJgKWDTM427+MqYbkQzD321DuCQJUqOuJKIA0IM2+W2xtYHdKOmZ4dR6fExsd4SXL+WQ==", + "dev": true + }, + "execa": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/execa/-/execa-5.1.1.tgz", + "integrity": "sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==", + "dev": true, + "requires": { + "cross-spawn": "^7.0.3", + "get-stream": "^6.0.0", + "human-signals": "^2.1.0", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.1", + "onetime": "^5.1.2", + "signal-exit": "^3.0.3", + "strip-final-newline": "^2.0.0" + }, + "dependencies": { + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, + "exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=", + "dev": true + }, + "expand-brackets": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", + "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", + "dev": true, + "requires": { + "debug": "^2.3.3", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "posix-character-classes": "^0.1.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "expect": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/expect/-/expect-27.5.1.tgz", + "integrity": "sha512-E1q5hSUG2AmYQwQJ041nvgpkODHQvB+RKlB4IYdru6uJsyFTRyZAP463M+1lINorwbqAmUggi6+WwkD8lCS/Dw==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "jest-get-type": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1" + } + }, + "extend": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", + "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", + "dev": true + }, + "extend-object": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/extend-object/-/extend-object-1.0.0.tgz", + "integrity": "sha1-QlFPhAFdE1bK9Rh5ad+yvBvaCCM=", + "dev": true + }, + "extend-shallow": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", + "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", + "dev": true, + "requires": { + "assign-symbols": "^1.0.0", + "is-extendable": "^1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4" + } + } + } + }, + "external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "requires": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + } + }, + "extglob": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", + "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", + "dev": true, + "requires": { + "array-unique": "^0.3.2", + "define-property": "^1.0.0", + "expand-brackets": "^2.1.4", + "extend-shallow": "^2.0.1", + "fragment-cache": "^0.2.1", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + } + } + }, + "extract-stack": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-1.0.0.tgz", + "integrity": "sha1-uXrK+UQe6iMyUpYktzL8WhyBZfo=", + "dev": true + }, + "extsprintf": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", + "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", + "dev": true + }, + "fancy-test": { + "version": "1.4.10", + "resolved": "https://registry.npmjs.org/fancy-test/-/fancy-test-1.4.10.tgz", + "integrity": "sha512-AaUX6wKS7D5OP2YK2q5G7c8PGx2lgoyLUD7Bbg8z323sb9aebBqzb9UN6phzI73UgO/ViihmNfOxF3kdfZLhew==", + "dev": true, + "requires": { + "@types/chai": "*", + "@types/lodash": "*", + "@types/node": "*", + "@types/sinon": "*", + "lodash": "^4.17.13", + "mock-stdin": "^1.0.0", + "nock": "^13.0.0", + "stdout-stderr": "^0.1.9" + } + }, + "fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "fast-glob": { + "version": "3.2.11", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.11.tgz", + "integrity": "sha512-xrO3+1bxSo3ZVHAnqzyuewYT6aMFHRAd4Kcs92MAonjwQZLsK9d0SF1IyQ3k5PoirxTW0Oe/RqFgMQ6TcNE5Ew==", + "dev": true, + "requires": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.2", + "merge2": "^1.3.0", + "micromatch": "^4.0.4" + } + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", + "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "dev": true + }, + "fast-levenshtein": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", + "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", + "dev": true + }, + "fast-xml-parser": { + "version": "3.21.1", + "resolved": "https://registry.npmjs.org/fast-xml-parser/-/fast-xml-parser-3.21.1.tgz", + "integrity": "sha512-FTFVjYoBOZTJekiUsawGsSYV9QL0A+zDYCRj7y34IO6Jg+2IMYEtQa+bbictpdpV8dHxXywqU7C0gRDEOFtBFg==", + "dev": true, + "requires": { + "strnum": "^1.0.4" + } + }, + "fastest-levenshtein": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/fastest-levenshtein/-/fastest-levenshtein-1.0.12.tgz", + "integrity": "sha512-On2N+BpYJ15xIC974QNVuYGMOlEVt4s0EOI3wwMqOmK1fdDY+FN/zltPV8vosq4ad4c/gJ1KHScUn/6AWIgiow==", + "dev": true + }, + "fastq": { + "version": "1.13.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.13.0.tgz", + "integrity": "sha512-YpkpUnK8od0o1hmeSc7UUs/eB/vIPWJYjKck2QKIzAf71Vm1AAQ3EbuZB3g2JIy+pg+ERD0vqI79KyZiB2e2Nw==", + "dev": true, + "requires": { + "reusify": "^1.0.4" + } + }, + "faye": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/faye/-/faye-1.4.0.tgz", + "integrity": "sha512-kRrIg4be8VNYhycS2PY//hpBJSzZPr/DBbcy9VWelhZMW3KhyLkQR0HL0k0MNpmVoNFF4EdfMFkNAWjTP65g6w==", + "dev": true, + "requires": { + "asap": "*", + "csprng": "*", + "faye-websocket": ">=0.9.1", + "safe-buffer": "*", + "tough-cookie": "*", + "tunnel-agent": "*" + } + }, + "faye-websocket": { + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.4.tgz", + "integrity": "sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==", + "dev": true, + "requires": { + "websocket-driver": ">=0.5.1" + } + }, + "fb-watchman": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.1.tgz", + "integrity": "sha512-DkPJKQeY6kKwmuMretBhr7G6Vodr7bFwDYTXIkfG1gjvNpaxBTQV3PbXg6bR1c1UP4jPOX0jHUbbHANL9vRjVg==", + "dev": true, + "requires": { + "bser": "2.1.1" + } + }, + "figures": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "requires": { + "escape-string-regexp": "^1.0.5" + } + }, + "file-entry-cache": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", + "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "dev": true, + "requires": { + "flat-cache": "^2.0.1" + } + }, + "file-set": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/file-set/-/file-set-4.0.2.tgz", + "integrity": "sha512-fuxEgzk4L8waGXaAkd8cMr73Pm0FxOVkn8hztzUW7BAHhOGH90viQNXbiOsnecCWmfInqU6YmAMwxRMdKETceQ==", + "dev": true, + "requires": { + "array-back": "^5.0.0", + "glob": "^7.1.6" + }, + "dependencies": { + "array-back": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-5.0.0.tgz", + "integrity": "sha512-kgVWwJReZWmVuWOQKEOohXKJX+nD02JAZ54D1RRWlv8L0NebauKAaFxACKzB74RTclt1+WNz5KHaLRDAPZbDEw==", + "dev": true + } + } + }, + "file-uri-to-path": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-2.0.0.tgz", + "integrity": "sha512-hjPFI8oE/2iQPVe4gbrJ73Pp+Xfub2+WI2LlXDbsaJBwT5wuMh35WNWVYYTpnz895shtwfyutMFLFywpQAFdLg==", + "dev": true + }, + "filelist": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.2.tgz", + "integrity": "sha512-z7O0IS8Plc39rTCq6i6iHxk43duYOn8uFJiWSewIq0Bww1RNybVHSCjahmcC87ZqAm4OTvFzlzeGu3XAzG1ctQ==", + "dev": true, + "requires": { + "minimatch": "^3.0.4" + } + }, + "filesize": { + "version": "6.4.0", + "resolved": "https://registry.npmjs.org/filesize/-/filesize-6.4.0.tgz", + "integrity": "sha512-mjFIpOHC4jbfcTfoh4rkWpI31mF7viw9ikj/JyLoKzqlwG/YsefKfvYlYhdYdg/9mtK2z1AzgN/0LvVQ3zdlSQ==", + "dev": true + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "find-java-home": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/find-java-home/-/find-java-home-1.1.0.tgz", + "integrity": "sha512-bSTCKNZ193UM/+ZZoNDzICAEHcVywovkhsWCkZALjCvRXQ+zXTe/XATrrP4CpxkaP6YFhQJOpyRpH0P2U/woDA==", + "dev": true, + "requires": { + "which": "~1.0.5", + "winreg": "~1.2.2" + }, + "dependencies": { + "which": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/which/-/which-1.0.9.tgz", + "integrity": "sha1-RgwdoPgQED0DIam2M6+eV15kSG8=", + "dev": true + } + } + }, + "find-replace": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-replace/-/find-replace-3.0.0.tgz", + "integrity": "sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ==", + "dev": true, + "requires": { + "array-back": "^3.0.1" + }, + "dependencies": { + "array-back": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz", + "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==", + "dev": true + } + } + }, + "find-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", + "dev": true, + "requires": { + "locate-path": "^6.0.0", + "path-exists": "^4.0.0" + } + }, + "first-chunk-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/first-chunk-stream/-/first-chunk-stream-2.0.0.tgz", + "integrity": "sha1-G97NuOCDwGZLkZRVgVd6Q6nzHXA=", + "dev": true, + "requires": { + "readable-stream": "^2.0.2" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } + } + }, + "flat-cache": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", + "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "dev": true, + "requires": { + "flatted": "^2.0.0", + "rimraf": "2.6.3", + "write": "1.0.3" + }, + "dependencies": { + "rimraf": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", + "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + } + } + }, + "flatted": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", + "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "dev": true + }, + "follow-redirects": { + "version": "1.14.8", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.14.8.tgz", + "integrity": "sha512-1x0S9UVJHsQprFcEC/qnNzBLcIxsjAV905f/UkQxbclCsoTWlacCNOpQa/anodLl2uaEKFhfWOvM2Qg77+15zA==" + }, + "for-in": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", + "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", + "dev": true + }, + "forever-agent": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", + "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", + "dev": true + }, + "form-data": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", + "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", + "dev": true, + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + } + }, + "fragment-cache": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", + "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", + "dev": true, + "requires": { + "map-cache": "^0.2.2" + } + }, + "from2": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", + "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "readable-stream": "^2.0.0" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } + } + }, + "fs-constants": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", + "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", + "dev": true + }, + "fs-extra": { + "version": "10.0.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.0.1.tgz", + "integrity": "sha512-NbdoVMZso2Lsrn/QwLXOy6rm0ufY2zEOKCDzJR/0kBsb0E6qed0P3iYK+Ath3BfvXEeu4JhEtXLgILx5psUfag==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "dependencies": { + "jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6", + "universalify": "^2.0.0" + } + }, + "universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true + } + } + }, + "fs-minipass": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz", + "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", + "dev": true, + "requires": { + "minipass": "^2.6.0" + } + }, + "fs-then-native": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/fs-then-native/-/fs-then-native-2.0.0.tgz", + "integrity": "sha1-GaEk2U2QwiyOBF8ujdbr6jbUjGc=", + "dev": true + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" + }, + "fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "optional": true + }, + "fstream": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/fstream/-/fstream-1.0.12.tgz", + "integrity": "sha512-WvJ193OHa0GHPEL+AycEJgxvBEwyfRkN1vhjca23OaPVMCaLCXTd5qAu82AjTcgP1UJmytkOKb63Ypde7raDIg==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "inherits": "~2.0.0", + "mkdirp": ">=0.5 0", + "rimraf": "2" + } + }, + "ftp": { + "version": "0.3.10", + "resolved": "https://registry.npmjs.org/ftp/-/ftp-0.3.10.tgz", + "integrity": "sha1-kZfYYa2BQvPmPVqDv+TFn3MwiF0=", + "dev": true, + "requires": { + "readable-stream": "1.1.x", + "xregexp": "2.0.0" + }, + "dependencies": { + "isarray": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-0.0.1.tgz", + "integrity": "sha1-ihis/Kmo9Bd+Cav8YDiTmwXR7t8=", + "dev": true + }, + "readable-stream": { + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-1.1.14.tgz", + "integrity": "sha1-fPTFTvZI44EwhMY23SB54WbAgdk=", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.1", + "isarray": "0.0.1", + "string_decoder": "~0.10.x" + } + }, + "string_decoder": { + "version": "0.10.31", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-0.10.31.tgz", + "integrity": "sha1-YuIDvEF2bGwoyfyEMB2rHFMQ+pQ=", + "dev": true + } + } + }, + "function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + }, + "functional-red-black-tree": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", + "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", + "dev": true + }, + "gensync": { + "version": "1.0.0-beta.2", + "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", + "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "dev": true + }, + "get-caller-file": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dev": true + }, + "get-intrinsic": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", + "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1" + } + }, + "get-package-type": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", + "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "dev": true + }, + "get-stdin": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/get-stdin/-/get-stdin-4.0.1.tgz", + "integrity": "sha1-uWjGsKBDhDJJAui/Gl3zJXmkUP4=", + "dev": true + }, + "get-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", + "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "dev": true + }, + "get-symbol-description": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz", + "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "get-intrinsic": "^1.1.1" + } + }, + "get-uri": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/get-uri/-/get-uri-3.0.2.tgz", + "integrity": "sha512-+5s0SJbGoyiJTZZ2JTpFPLMPSch72KEqGOTvQsBqg0RBWvwhWUSYZFAtz3TPW0GXJuLBJPts1E241iHg+VRfhg==", + "dev": true, + "requires": { + "@tootallnate/once": "1", + "data-uri-to-buffer": "3", + "debug": "4", + "file-uri-to-path": "2", + "fs-extra": "^8.1.0", + "ftp": "^0.3.10" + }, + "dependencies": { + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + } + } + }, + "get-value": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", + "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", + "dev": true + }, + "getpass": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", + "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", + "dev": true, + "requires": { + "assert-plus": "^1.0.0" + } + }, + "gh-got": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/gh-got/-/gh-got-5.0.0.tgz", + "integrity": "sha1-7pW+NxBv2HSKlvjR20uuqJ4b+oo=", + "dev": true, + "requires": { + "got": "^6.2.0", + "is-plain-obj": "^1.1.0" + } + }, + "github-username": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/github-username/-/github-username-3.0.0.tgz", + "integrity": "sha1-CnciGbMTB0NCnyRW0L3T21Xc57E=", + "dev": true, + "requires": { + "gh-got": "^5.0.0" + } + }, + "glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "requires": { + "is-glob": "^4.0.1" + } + }, + "glob-to-regexp": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", + "integrity": "sha1-jFoUlNIGbFcMw7/kSWF1rMTVAqs=", + "dev": true + }, + "globals": { + "version": "11.12.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", + "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "dev": true + }, + "globby": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.1.0.tgz", + "integrity": "sha512-jhIXaOzy1sb8IyocaruWSn1TjmnBVs8Ayhcy83rmxNJ8q2uWKCAj3CnJY+KpGSXCueAPc0i05kVvVKtP1t9S3g==", + "dev": true, + "requires": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.2.9", + "ignore": "^5.2.0", + "merge2": "^1.4.1", + "slash": "^3.0.0" + } + }, + "got": { + "version": "6.7.1", + "resolved": "https://registry.npmjs.org/got/-/got-6.7.1.tgz", + "integrity": "sha1-JAzQV4WpoY5WHcG0S0HHY+8ejbA=", + "dev": true, + "requires": { + "create-error-class": "^3.0.0", + "duplexer3": "^0.1.4", + "get-stream": "^3.0.0", + "is-redirect": "^1.0.0", + "is-retry-allowed": "^1.0.0", + "is-stream": "^1.0.0", + "lowercase-keys": "^1.0.0", + "safe-buffer": "^5.0.1", + "timed-out": "^4.0.0", + "unzip-response": "^2.0.1", + "url-parse-lax": "^1.0.0" + }, + "dependencies": { + "get-stream": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", + "dev": true + }, + "is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", + "dev": true + } + } + }, + "graceful-fs": { + "version": "4.2.10", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", + "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", + "dev": true + }, + "grouped-queue": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/grouped-queue/-/grouped-queue-1.1.0.tgz", + "integrity": "sha512-rZOFKfCqLhsu5VqjBjEWiwrYqJR07KxIkH4mLZlNlGDfntbb4FbMyGFP14TlvRPrU9S3Hnn/sgxbC5ZeN0no3Q==", + "dev": true, + "requires": { + "lodash": "^4.17.15" + } + }, + "handlebars": { + "version": "4.7.7", + "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.7.tgz", + "integrity": "sha512-aAcXm5OAfE/8IXkcZvCepKU3VzW1/39Fb5ZuqMtgI/hT8X2YgoMvBY5dLhq/cpOvw7Lk1nK/UF71aLG/ZnVYRA==", + "dev": true, + "requires": { + "minimist": "^1.2.5", + "neo-async": "^2.6.0", + "source-map": "^0.6.1", + "uglify-js": "^3.1.4", + "wordwrap": "^1.0.0" + }, + "dependencies": { + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true + } + } + }, + "har-schema": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", + "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", + "dev": true + }, + "har-validator": { + "version": "5.1.5", + "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", + "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", + "dev": true, + "requires": { + "ajv": "^6.12.3", + "har-schema": "^2.0.0" + } + }, + "has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "requires": { + "function-bind": "^1.1.1" + } + }, + "has-ansi": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", + "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + } + } + }, + "has-bigints": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.1.tgz", + "integrity": "sha512-LSBS2LjbNBTf6287JEbEzvJgftkF5qFkmCo9hDRpAzKhUOlJ+hx8dd4USs00SgsUNwc4617J9ki5YtEClM2ffA==", + "dev": true + }, + "has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "dev": true + }, + "has-symbol-support-x": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/has-symbol-support-x/-/has-symbol-support-x-1.4.2.tgz", + "integrity": "sha512-3ToOva++HaW+eCpgqZrCfN51IPB+7bJNVT6CUATzueB5Heb8o6Nam0V3HG5dlDvZU1Gn5QLcbahiKw/XVk5JJw==", + "dev": true + }, + "has-symbols": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.3.tgz", + "integrity": "sha512-l3LCuF6MgDNwTDKkdYGEihYjt5pRPbEg46rtlmnSPlUbgmB8LOIrKJbYYFBSbnPaJexMKtiPO8hmeRjRz2Td+A==", + "dev": true + }, + "has-to-string-tag-x": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/has-to-string-tag-x/-/has-to-string-tag-x-1.4.1.tgz", + "integrity": "sha512-vdbKfmw+3LoOYVr+mtxHaX5a96+0f3DljYd8JOqvOLsf5mw2Otda2qCDT9qRqLAhrjyQ0h7ual5nOiASpsGNFw==", + "dev": true, + "requires": { + "has-symbol-support-x": "^1.4.1" + } + }, + "has-tostringtag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz", + "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", + "dev": true, + "requires": { + "has-symbols": "^1.0.2" + } + }, + "has-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", + "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", + "dev": true, + "requires": { + "get-value": "^2.0.6", + "has-values": "^1.0.0", + "isobject": "^3.0.0" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "has-values": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", + "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "kind-of": "^4.0.0" + }, + "dependencies": { + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "kind-of": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", + "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "he": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", + "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", + "dev": true + }, + "header-case": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/header-case/-/header-case-2.0.4.tgz", + "integrity": "sha512-H/vuk5TEEVZwrR0lp2zed9OCo1uAILMlx0JEMgC26rzyJJ3N1v6XkwHHXJQdR2doSjcGPM6OKPYoJgf0plJ11Q==", + "dev": true, + "requires": { + "capital-case": "^1.0.4", + "tslib": "^2.0.3" + } + }, + "heroku-cli-util": { + "version": "8.0.12", + "resolved": "https://registry.npmjs.org/heroku-cli-util/-/heroku-cli-util-8.0.12.tgz", + "integrity": "sha512-63wB17oSktlA/HzpIV/PGe8Isq5AZArT51KAW1gg54zyYRIiHOwXik93HZuuRVUaVrWvVUhItFeLgqMwAwlTsw==", + "dev": true, + "requires": { + "@heroku-cli/color": "^1.1.3", + "ansi-escapes": "^3.1.0", + "ansi-styles": "^3.2.1", + "cardinal": "^2.0.1", + "chalk": "^2.4.1", + "co": "^4.6.0", + "got": "^8.3.1", + "heroku-client": "^3.1.0", + "lodash": "^4.17.10", + "netrc-parser": "^3.1.4", + "opn": "^3.0.3", + "strip-ansi": "^4.0.0", + "supports-color": "^5.4.0", + "tslib": "^1.9.0", + "tunnel-agent": "^0.6.0" + }, + "dependencies": { + "@sindresorhus/is": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/@sindresorhus/is/-/is-0.7.0.tgz", + "integrity": "sha512-ONhaKPIufzzrlNbqtWFFd+jlnemX6lJAgq9ZeiZtS7I1PIf/la7CW4m83rTXRnVnsMbW2k56pGYu7AUFJD9Pow==", + "dev": true + }, + "ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true + }, + "cacheable-request": { + "version": "2.1.4", + "resolved": "https://registry.npmjs.org/cacheable-request/-/cacheable-request-2.1.4.tgz", + "integrity": "sha1-DYCIAbY0KtM8kd+dC0TcCbkeXD0=", + "dev": true, + "requires": { + "clone-response": "1.0.2", + "get-stream": "3.0.0", + "http-cache-semantics": "3.8.1", + "keyv": "3.0.0", + "lowercase-keys": "1.0.0", + "normalize-url": "2.0.1", + "responselike": "1.0.2" + }, + "dependencies": { + "lowercase-keys": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.0.tgz", + "integrity": "sha1-TjNms55/VFfjXxMkvfb4jQv8cwY=", + "dev": true + } + } + }, + "decompress-response": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-3.3.0.tgz", + "integrity": "sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M=", + "dev": true, + "requires": { + "mimic-response": "^1.0.0" + } + }, + "get-stream": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", + "dev": true + }, + "got": { + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/got/-/got-8.3.2.tgz", + "integrity": "sha512-qjUJ5U/hawxosMryILofZCkm3C84PLJS/0grRIpjAwu+Lkxxj5cxeCU25BG0/3mDSpXKTyZr8oh8wIgLaH0QCw==", + "dev": true, + "requires": { + "@sindresorhus/is": "^0.7.0", + "cacheable-request": "^2.1.1", + "decompress-response": "^3.3.0", + "duplexer3": "^0.1.4", + "get-stream": "^3.0.0", + "into-stream": "^3.1.0", + "is-retry-allowed": "^1.1.0", + "isurl": "^1.0.0-alpha5", + "lowercase-keys": "^1.0.0", + "mimic-response": "^1.0.0", + "p-cancelable": "^0.4.0", + "p-timeout": "^2.0.1", + "pify": "^3.0.0", + "safe-buffer": "^5.1.1", + "timed-out": "^4.0.1", + "url-parse-lax": "^3.0.0", + "url-to-options": "^1.0.1" + } + }, + "http-cache-semantics": { + "version": "3.8.1", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-3.8.1.tgz", + "integrity": "sha512-5ai2iksyV8ZXmnZhHH4rWPoxxistEexSi5936zIQ1bnNTW5VnA85B6P/VpXiRM017IgRvb2kKo1a//y+0wSp3w==", + "dev": true + }, + "json-buffer": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.0.tgz", + "integrity": "sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg=", + "dev": true + }, + "keyv": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-3.0.0.tgz", + "integrity": "sha512-eguHnq22OE3uVoSYG0LVWNP+4ppamWr9+zWBe1bsNcovIMy6huUJFPgy4mGwCd/rnl3vOLGW1MTlu4c57CT1xA==", + "dev": true, + "requires": { + "json-buffer": "3.0.0" + } + }, + "normalize-url": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-2.0.1.tgz", + "integrity": "sha512-D6MUW4K/VzoJ4rJ01JFKxDrtY1v9wrgzCX5f2qj/lzH1m/lW6MhUZFKerVsnyjOhOsYzI9Kqqak+10l4LvLpMw==", + "dev": true, + "requires": { + "prepend-http": "^2.0.0", + "query-string": "^5.0.1", + "sort-keys": "^2.0.0" + } + }, + "opn": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/opn/-/opn-3.0.3.tgz", + "integrity": "sha1-ttmec5n3jWXDuq/+8fsojpuFJDo=", + "dev": true, + "requires": { + "object-assign": "^4.0.1" + } + }, + "p-cancelable": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-0.4.1.tgz", + "integrity": "sha512-HNa1A8LvB1kie7cERyy21VNeHb2CWJJYqyyC2o3klWFfMGlFmWv2Z7sFgZH8ZiaYL95ydToKTFVXgMV/Os0bBQ==", + "dev": true + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "prepend-http": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-2.0.0.tgz", + "integrity": "sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc=", + "dev": true + }, + "responselike": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/responselike/-/responselike-1.0.2.tgz", + "integrity": "sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec=", + "dev": true, + "requires": { + "lowercase-keys": "^1.0.0" + } + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + }, + "tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "url-parse-lax": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-3.0.0.tgz", + "integrity": "sha1-FrXK/Afb42dsGxmZF3gj1lA6yww=", + "dev": true, + "requires": { + "prepend-http": "^2.0.0" + } + } + } + }, + "heroku-client": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/heroku-client/-/heroku-client-3.1.0.tgz", + "integrity": "sha512-UfGKwUm5duzzSVI8uUXlNAE1mus6uPxmZPji4vuG1ArV5DYL1rXsZShp0OoxraWdEwYoxCUrM6KGztC68x5EZQ==", + "dev": true, + "requires": { + "is-retry-allowed": "^1.0.0", + "tunnel-agent": "^0.6.0" + } + }, + "hosted-git-info": { + "version": "2.8.9", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz", + "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", + "dev": true + }, + "html-encoding-sniffer": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-2.0.1.tgz", + "integrity": "sha512-D5JbOMBIR/TVZkubHT+OyT2705QvogUW4IBn6nHd756OwieSF9aDYFj4dv6HHEVGYbHaLETa3WggZYWWMyy3ZQ==", + "dev": true, + "requires": { + "whatwg-encoding": "^1.0.5" + } + }, + "html-entities": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-2.3.3.tgz", + "integrity": "sha512-DV5Ln36z34NNTDgnz0EWGBLZENelNAtkiFA4kyNOG2tDI6Mz1uSWiq1wAKdyjnJwyDiDO7Fa2SO1CTxPXL8VxA==", + "dev": true + }, + "html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true + }, + "http-cache-semantics": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", + "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", + "dev": true + }, + "http-call": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/http-call/-/http-call-5.3.0.tgz", + "integrity": "sha512-ahwimsC23ICE4kPl9xTBjKB4inbRaeLyZeRunC/1Jy/Z6X8tv22MEAjK+KBOMSVLaqXPTTmd8638waVIKLGx2w==", + "dev": true, + "requires": { + "content-type": "^1.0.4", + "debug": "^4.1.1", + "is-retry-allowed": "^1.1.0", + "is-stream": "^2.0.0", + "parse-json": "^4.0.0", + "tunnel-agent": "^0.6.0" + }, + "dependencies": { + "parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "dev": true, + "requires": { + "error-ex": "^1.3.1", + "json-parse-better-errors": "^1.0.1" + } + } + } + }, + "http-errors": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz", + "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==", + "dev": true, + "requires": { + "depd": "2.0.0", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": "2.0.1", + "toidentifier": "1.0.1" + } + }, + "http-parser-js": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.6.tgz", + "integrity": "sha512-vDlkRPDJn93swjcjqMSaGSPABbIarsr1TLAui/gLDXzV5VsJNdXNzMYDyNBLQkjWQCJ1uizu8T2oDMhmGt0PRA==", + "dev": true + }, + "http-proxy-agent": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", + "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", + "dev": true, + "requires": { + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" + } + }, + "http-signature": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", + "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", + "dev": true, + "requires": { + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" + } + }, + "http2-wrapper": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/http2-wrapper/-/http2-wrapper-1.0.3.tgz", + "integrity": "sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg==", + "dev": true, + "requires": { + "quick-lru": "^5.1.1", + "resolve-alpn": "^1.0.0" + } + }, + "https-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", + "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", + "dev": true, + "requires": { + "agent-base": "6", + "debug": "4" + } + }, + "human-signals": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", + "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", + "dev": true + }, + "husky": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/husky/-/husky-7.0.4.tgz", + "integrity": "sha512-vbaCKN2QLtP/vD4yvs6iz6hBEo6wkSzs8HpRah1Z6aGmF2KW5PdYuAd7uX5a+OyBZHBhd+TFLqgjUgytQr4RvQ==", + "dev": true + }, + "hyperlinker": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/hyperlinker/-/hyperlinker-1.0.0.tgz", + "integrity": "sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ==", + "dev": true + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "ieee754": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", + "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", + "dev": true + }, + "ignore": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.0.tgz", + "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", + "dev": true + }, + "import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "requires": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "dependencies": { + "resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true + } + } + }, + "import-local": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.1.0.tgz", + "integrity": "sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==", + "dev": true, + "requires": { + "pkg-dir": "^4.2.0", + "resolve-cwd": "^3.0.0" + } + }, + "imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true + }, + "indent-string": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", + "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", + "dev": true + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "ini": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", + "dev": true + }, + "inquirer": { + "version": "7.3.3", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.3.3.tgz", + "integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==", + "dev": true, + "requires": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.19", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.6.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "internal-slot": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.3.tgz", + "integrity": "sha512-O0DB1JC/sPyZl7cIo78n5dR7eUSwwpYPiXRhTzNxZVAMUuB8vlnRFyLxdrVToks6XPLVnFfbzaVd5WLjhgg+vA==", + "dev": true, + "requires": { + "get-intrinsic": "^1.1.0", + "has": "^1.0.3", + "side-channel": "^1.0.4" + } + }, + "interpret": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.4.0.tgz", + "integrity": "sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==" + }, + "into-stream": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/into-stream/-/into-stream-3.1.0.tgz", + "integrity": "sha1-lvsKk2wSur1v8XUqF9BWFqvQlMY=", + "dev": true, + "requires": { + "from2": "^2.1.1", + "p-is-promise": "^1.1.0" + } + }, + "ip": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/ip/-/ip-1.1.5.tgz", + "integrity": "sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo=", + "dev": true + }, + "is-accessor-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", + "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-arrayish": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", + "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", + "dev": true + }, + "is-bigint": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", + "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", + "dev": true, + "requires": { + "has-bigints": "^1.0.1" + } + }, + "is-boolean-object": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz", + "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, + "is-buffer": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", + "dev": true + }, + "is-callable": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.4.tgz", + "integrity": "sha512-nsuwtxZfMX67Oryl9LCQ+upnC0Z0BgpwntpS89m1H/TLF0zNfzfLMV/9Wa/6MZsj0acpEjAO0KF1xT6ZdLl95w==", + "dev": true + }, + "is-core-module": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.8.0.tgz", + "integrity": "sha512-vd15qHsaqrRL7dtH6QNuy0ndJmRDrS9HAM1CAiSifNUFv4x1a0CCVsj18hJ1mShxIG6T2i1sO78MkP56r0nYRw==", + "requires": { + "has": "^1.0.3" + } + }, + "is-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "is-date-object": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", + "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-descriptor": { + "version": "0.1.6", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", + "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "dependencies": { + "kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true + } + } + }, + "is-docker": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", + "dev": true + }, + "is-extendable": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", + "dev": true + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true + }, + "is-generator-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", + "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "dev": true + }, + "is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-negative-zero": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", + "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "dev": true + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true + }, + "is-number-object": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.7.tgz", + "integrity": "sha512-k1U0IRzLMo7ZlYIfzRu23Oh6MiIFasgpb9X76eqfFZAqwH44UI4KTBvBYIZ1dSL9ZzChTB9ShHfLkR4pdW5krQ==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-object": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-object/-/is-object-1.0.2.tgz", + "integrity": "sha512-2rRIahhZr2UWb45fIOuvZGpFtz0TyOZLf32KxBbSoUCeZR495zCKlWUKKUByk3geS2eAs7ZAABt0Y/Rx0GiQGA==", + "dev": true + }, + "is-plain-obj": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", + "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=", + "dev": true + }, + "is-plain-object": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", + "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", + "dev": true, + "requires": { + "isobject": "^3.0.1" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "is-potential-custom-element-name": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", + "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==", + "dev": true + }, + "is-redirect": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-redirect/-/is-redirect-1.0.0.tgz", + "integrity": "sha1-HQPd7VO9jbDzDCbk+V02/HyH3CQ=", + "dev": true + }, + "is-regex": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", + "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "has-tostringtag": "^1.0.0" + } + }, + "is-retry-allowed": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/is-retry-allowed/-/is-retry-allowed-1.2.0.tgz", + "integrity": "sha512-RUbUeKwvm3XG2VYamhJL1xFktgjvPzL0Hq8C+6yrWIswDy3BIXGqCxhxkc30N9jqK311gVU137K8Ei55/zVJRg==", + "dev": true + }, + "is-scoped": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-scoped/-/is-scoped-1.0.0.tgz", + "integrity": "sha1-RJypgpnnEwOCViieyytUDcQ3yzA=", + "dev": true, + "requires": { + "scoped-regex": "^1.0.0" + } + }, + "is-shared-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", + "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-stream": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", + "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", + "dev": true + }, + "is-string": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", + "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "dev": true, + "requires": { + "has-tostringtag": "^1.0.0" + } + }, + "is-symbol": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", + "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "dev": true, + "requires": { + "has-symbols": "^1.0.2" + } + }, + "is-typedarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", + "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", + "dev": true + }, + "is-utf8": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/is-utf8/-/is-utf8-0.2.1.tgz", + "integrity": "sha1-Sw2hRCEE0bM2NA6AeX6GXPOffXI=", + "dev": true + }, + "is-weakref": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", + "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2" + } + }, + "is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true + }, + "is-wsl": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "dev": true, + "requires": { + "is-docker": "^2.0.0" + } + }, + "is-zip": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-zip/-/is-zip-1.0.0.tgz", + "integrity": "sha1-R7Co/004p2QxzP2ZqOFaTIa6IyU=", + "dev": true + }, + "isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "dev": true + }, + "isbinaryfile": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/isbinaryfile/-/isbinaryfile-4.0.10.tgz", + "integrity": "sha512-iHrqe5shvBUcFbmZq9zOQHBoeOhZJu6RQGrDpBgenUm/Am+F3JM2MgQj+rK3Z601fzrL5gLZWtAPH2OBaSVcyw==", + "dev": true + }, + "isexe": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", + "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "dev": true + }, + "isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "dev": true, + "requires": { + "isarray": "1.0.0" + } + }, + "isomorphic-git": { + "version": "1.17.0", + "resolved": "https://registry.npmjs.org/isomorphic-git/-/isomorphic-git-1.17.0.tgz", + "integrity": "sha512-8ToEVqYLeTE1Ys3UQ21yAxQf0rW7GYRvsENhvXNDONAHgNks1fsgUJH3mVzgbsGf4LpW3kuJI6e/e3VIeaTW3w==", + "dev": true, + "requires": { + "async-lock": "^1.1.0", + "clean-git-ref": "^2.0.1", + "crc-32": "^1.2.0", + "diff3": "0.0.3", + "ignore": "^5.1.4", + "minimisted": "^2.0.0", + "pako": "^1.0.10", + "pify": "^4.0.1", + "readable-stream": "^3.4.0", + "sha.js": "^2.4.9", + "simple-get": "^4.0.1" + }, + "dependencies": { + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + } + } + }, + "isstream": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", + "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", + "dev": true + }, + "istanbul-lib-coverage": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.0.tgz", + "integrity": "sha512-eOeJ5BHCmHYvQK7xt9GkdHuzuCGS1Y6g9Gvnx3Ym33fz/HpLRYxiS0wHNr+m/MBC8B647Xt608vCDEvhl9c6Mw==", + "dev": true + }, + "istanbul-lib-instrument": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-5.1.0.tgz", + "integrity": "sha512-czwUz525rkOFDJxfKK6mYfIs9zBKILyrZQxjz3ABhjQXhbhFsSbo1HW/BFcsDnfJYJWA6thRR5/TUY2qs5W99Q==", + "dev": true, + "requires": { + "@babel/core": "^7.12.3", + "@babel/parser": "^7.14.7", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.2.0", + "semver": "^6.3.0" + } + }, + "istanbul-lib-report": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", + "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", + "dev": true, + "requires": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^3.0.0", + "supports-color": "^7.1.0" + }, + "dependencies": { + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "istanbul-lib-source-maps": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", + "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "dev": true, + "requires": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^3.0.0", + "source-map": "^0.6.1" + }, + "dependencies": { + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true + } + } + }, + "istanbul-reports": { + "version": "3.1.4", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.4.tgz", + "integrity": "sha512-r1/DshN4KSE7xWEknZLLLLDn5CJybV3nw01VTkp6D5jzLuELlcbudfj/eSQFvrKsJuTVCGnePO7ho82Nw9zzfw==", + "dev": true, + "requires": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + } + }, + "istextorbinary": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/istextorbinary/-/istextorbinary-2.6.0.tgz", + "integrity": "sha512-+XRlFseT8B3L9KyjxxLjfXSLMuErKDsd8DBNrsaxoViABMEZlOSCstwmw0qpoFX3+U6yWU1yhLudAe6/lETGGA==", + "dev": true, + "requires": { + "binaryextensions": "^2.1.2", + "editions": "^2.2.0", + "textextensions": "^2.5.0" + } + }, + "isurl": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isurl/-/isurl-1.0.0.tgz", + "integrity": "sha512-1P/yWsxPlDtn7QeRD+ULKQPaIaN6yF368GZ2vDfv0AL0NwpStafjWCDDdn0k8wgFMWpVAqG7oJhxHnlud42i9w==", + "dev": true, + "requires": { + "has-to-string-tag-x": "^1.2.0", + "is-object": "^1.0.1" + } + }, + "jake": { + "version": "10.8.4", + "resolved": "https://registry.npmjs.org/jake/-/jake-10.8.4.tgz", + "integrity": "sha512-MtWeTkl1qGsWUtbl/Jsca/8xSoK3x0UmS82sNbjqxxG/de/M/3b1DntdjHgPMC50enlTNwXOCRqPXLLt5cCfZA==", + "dev": true, + "requires": { + "async": "0.9.x", + "chalk": "^4.0.2", + "filelist": "^1.0.1", + "minimatch": "^3.0.4" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "async": { + "version": "0.9.2", + "resolved": "https://registry.npmjs.org/async/-/async-0.9.2.tgz", + "integrity": "sha1-rqdNXmHB+JlhO/ZL2mbUx48v0X0=", + "dev": true + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest": { + "version": "27.0.6", + "resolved": "https://registry.npmjs.org/jest/-/jest-27.0.6.tgz", + "integrity": "sha512-EjV8aETrsD0wHl7CKMibKwQNQc3gIRBXlTikBmmHUeVMKaPFxdcUIBfoDqTSXDoGJIivAYGqCWVlzCSaVjPQsA==", + "dev": true, + "requires": { + "@jest/core": "^27.0.6", + "import-local": "^3.0.2", + "jest-cli": "^27.0.6" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "jest-cli": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-27.5.1.tgz", + "integrity": "sha512-Hc6HOOwYq4/74/c62dEE3r5elx8wjYqxY0r0G/nFrLDPMFRu6RA/u8qINOIkvhxG7mMQ5EJsOGfRpI8L6eFUVw==", + "dev": true, + "requires": { + "@jest/core": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "exit": "^0.1.2", + "graceful-fs": "^4.2.9", + "import-local": "^3.0.2", + "jest-config": "^27.5.1", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "prompts": "^2.0.1", + "yargs": "^16.2.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-changed-files": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-27.5.1.tgz", + "integrity": "sha512-buBLMiByfWGCoMsLLzGUUSpAmIAGnbR2KJoMN10ziLhOLvP4e0SlypHnAel8iqQXTrcbmfEY9sSqae5sgUsTvw==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "execa": "^5.0.0", + "throat": "^6.0.1" + } + }, + "jest-circus": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-27.5.1.tgz", + "integrity": "sha512-D95R7x5UtlMA5iBYsOHFFbMD/GVA4R/Kdq15f7xYWUfWHBto9NYRsOvnSauTgdF+ogCpJ4tyKOXhUifxS65gdw==", + "dev": true, + "requires": { + "@jest/environment": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "dedent": "^0.7.0", + "expect": "^27.5.1", + "is-generator-fn": "^2.0.0", + "jest-each": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "pretty-format": "^27.5.1", + "slash": "^3.0.0", + "stack-utils": "^2.0.3", + "throat": "^6.0.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-config": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-27.5.1.tgz", + "integrity": "sha512-5sAsjm6tGdsVbW9ahcChPAFCk4IlkQUknH5AvKjuLTSlcO/wCZKyFdn7Rg0EkC+OGgWODEy2hDpWB1PgzH0JNA==", + "dev": true, + "requires": { + "@babel/core": "^7.8.0", + "@jest/test-sequencer": "^27.5.1", + "@jest/types": "^27.5.1", + "babel-jest": "^27.5.1", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "deepmerge": "^4.2.2", + "glob": "^7.1.1", + "graceful-fs": "^4.2.9", + "jest-circus": "^27.5.1", + "jest-environment-jsdom": "^27.5.1", + "jest-environment-node": "^27.5.1", + "jest-get-type": "^27.5.1", + "jest-jasmine2": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-runner": "^27.5.1", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "micromatch": "^4.0.4", + "parse-json": "^5.2.0", + "pretty-format": "^27.5.1", + "slash": "^3.0.0", + "strip-json-comments": "^3.1.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-diff": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-27.5.1.tgz", + "integrity": "sha512-m0NvkX55LDt9T4mctTEgnZk3fmEg3NRYutvMPWM/0iPnkFj2wIeF45O1718cMSOFO1vINkqmxqD8vE37uTEbqw==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "diff-sequences": "^27.5.1", + "jest-get-type": "^27.5.1", + "pretty-format": "^27.5.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-docblock": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-27.5.1.tgz", + "integrity": "sha512-rl7hlABeTsRYxKiUfpHrQrG4e2obOiTQWfMEH3PxPjOtdsfLQO4ReWSZaQ7DETm4xu07rl4q/h4zcKXyU0/OzQ==", + "dev": true, + "requires": { + "detect-newline": "^3.0.0" + } + }, + "jest-each": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-27.5.1.tgz", + "integrity": "sha512-1Ff6p+FbhT/bXQnEouYy00bkNSY7OUpfIcmdl8vZ31A1UUaurOLPA8a8BbJOF2RDUElwJhmeaV7LnagI+5UwNQ==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "jest-get-type": "^27.5.1", + "jest-util": "^27.5.1", + "pretty-format": "^27.5.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-environment-jsdom": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-environment-jsdom/-/jest-environment-jsdom-27.5.1.tgz", + "integrity": "sha512-TFBvkTC1Hnnnrka/fUb56atfDtJ9VMZ94JkjTbggl1PEpwrYtUBKMezB3inLmWqQsXYLcMwNoDQwoBTAvFfsfw==", + "dev": true, + "requires": { + "@jest/environment": "^27.5.1", + "@jest/fake-timers": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "jest-mock": "^27.5.1", + "jest-util": "^27.5.1", + "jsdom": "^16.6.0" + } + }, + "jest-environment-node": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-27.5.1.tgz", + "integrity": "sha512-Jt4ZUnxdOsTGwSRAfKEnE6BcwsSPNOijjwifq5sDFSA2kesnXTvNqKHYgM0hDq3549Uf/KzdXNYn4wMZJPlFLw==", + "dev": true, + "requires": { + "@jest/environment": "^27.5.1", + "@jest/fake-timers": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "jest-mock": "^27.5.1", + "jest-util": "^27.5.1" + } + }, + "jest-get-type": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-27.5.1.tgz", + "integrity": "sha512-2KY95ksYSaK7DMBWQn6dQz3kqAf3BB64y2udeG+hv4KfSOb9qwcYQstTJc1KCbsix+wLZWZYN8t7nwX3GOBLRw==", + "dev": true + }, + "jest-haste-map": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-27.5.1.tgz", + "integrity": "sha512-7GgkZ4Fw4NFbMSDSpZwXeBiIbx+t/46nJ2QitkOjvwPYyZmqttu2TDSimMHP1EkPOi4xUZAN1doE5Vd25H4Jng==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "@types/graceful-fs": "^4.1.2", + "@types/node": "*", + "anymatch": "^3.0.3", + "fb-watchman": "^2.0.0", + "fsevents": "^2.3.2", + "graceful-fs": "^4.2.9", + "jest-regex-util": "^27.5.1", + "jest-serializer": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", + "micromatch": "^4.0.4", + "walker": "^1.0.7" + } + }, + "jest-jasmine2": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-jasmine2/-/jest-jasmine2-27.5.1.tgz", + "integrity": "sha512-jtq7VVyG8SqAorDpApwiJJImd0V2wv1xzdheGHRGyuT7gZm6gG47QEskOlzsN1PG/6WNaCo5pmwMHDf3AkG2pQ==", + "dev": true, + "requires": { + "@jest/environment": "^27.5.1", + "@jest/source-map": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "co": "^4.6.0", + "expect": "^27.5.1", + "is-generator-fn": "^2.0.0", + "jest-each": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "pretty-format": "^27.5.1", + "throat": "^6.0.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-leak-detector": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-27.5.1.tgz", + "integrity": "sha512-POXfWAMvfU6WMUXftV4HolnJfnPOGEu10fscNCA76KBpRRhcMN2c8d3iT2pxQS3HLbA+5X4sOUPzYO2NUyIlHQ==", + "dev": true, + "requires": { + "jest-get-type": "^27.5.1", + "pretty-format": "^27.5.1" + } + }, + "jest-matcher-utils": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-27.5.1.tgz", + "integrity": "sha512-z2uTx/T6LBaCoNWNFWwChLBKYxTMcGBRjAt+2SbP929/Fflb9aa5LGma654Rz8z9HLxsrUaYzxE9T/EFIL/PAw==", + "dev": true, + "requires": { + "chalk": "^4.0.0", + "jest-diff": "^27.5.1", + "jest-get-type": "^27.5.1", + "pretty-format": "^27.5.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-message-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-27.5.1.tgz", + "integrity": "sha512-rMyFe1+jnyAAf+NHwTclDz0eAaLkVDdKVHHBFWsBWHnnh5YeJMNWWsv7AbFYXfK3oTqvL7VTWkhNLu1jX24D+g==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.12.13", + "@jest/types": "^27.5.1", + "@types/stack-utils": "^2.0.0", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "micromatch": "^4.0.4", + "pretty-format": "^27.5.1", + "slash": "^3.0.0", + "stack-utils": "^2.0.3" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-mock": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-27.5.1.tgz", + "integrity": "sha512-K4jKbY1d4ENhbrG2zuPWaQBvDly+iZ2yAW+T1fATN78hc0sInwn7wZB8XtlNnvHug5RMwV897Xm4LqmPM4e2Og==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "@types/node": "*" + } + }, + "jest-pnp-resolver": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.2.tgz", + "integrity": "sha512-olV41bKSMm8BdnuMsewT4jqlZ8+3TCARAXjZGT9jcoSnrfUnRCqnMoF9XEeoWjbzObpqF9dRhHQj0Xb9QdF6/w==", + "dev": true + }, + "jest-regex-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-27.5.1.tgz", + "integrity": "sha512-4bfKq2zie+x16okqDXjXn9ql2B0dScQu+vcwe4TvFVhkVyuWLqpZrZtXxLLWoXYgn0E87I6r6GRYHF7wFZBUvg==", + "dev": true + }, + "jest-resolve": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-27.5.1.tgz", + "integrity": "sha512-FFDy8/9E6CV83IMbDpcjOhumAQPDyETnU2KZ1O98DwTnz8AOBsW/Xv3GySr1mOZdItLR+zDZ7I/UdTFbgSOVCw==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-pnp-resolver": "^1.2.2", + "jest-util": "^27.5.1", + "jest-validate": "^27.5.1", + "resolve": "^1.20.0", + "resolve.exports": "^1.1.0", + "slash": "^3.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-resolve-dependencies": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-27.5.1.tgz", + "integrity": "sha512-QQOOdY4PE39iawDn5rzbIePNigfe5B9Z91GDD1ae/xNDlu9kaat8QQ5EKnNmVWPV54hUdxCVwwj6YMgR2O7IOg==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-snapshot": "^27.5.1" + } + }, + "jest-runner": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-27.5.1.tgz", + "integrity": "sha512-g4NPsM4mFCOwFKXO4p/H/kWGdJp9V8kURY2lX8Me2drgXqG7rrZAx5kv+5H7wtt/cdFIjhqYx1HrlqWHaOvDaQ==", + "dev": true, + "requires": { + "@jest/console": "^27.5.1", + "@jest/environment": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "emittery": "^0.8.1", + "graceful-fs": "^4.2.9", + "jest-docblock": "^27.5.1", + "jest-environment-jsdom": "^27.5.1", + "jest-environment-node": "^27.5.1", + "jest-haste-map": "^27.5.1", + "jest-leak-detector": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-runtime": "^27.5.1", + "jest-util": "^27.5.1", + "jest-worker": "^27.5.1", + "source-map-support": "^0.5.6", + "throat": "^6.0.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-runtime": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-27.5.1.tgz", + "integrity": "sha512-o7gxw3Gf+H2IGt8fv0RiyE1+r83FJBRruoA+FXrlHw6xEyBsU8ugA6IPfTdVyA0w8HClpbK+DGJxH59UrNMx8A==", + "dev": true, + "requires": { + "@jest/environment": "^27.5.1", + "@jest/fake-timers": "^27.5.1", + "@jest/globals": "^27.5.1", + "@jest/source-map": "^27.5.1", + "@jest/test-result": "^27.5.1", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "chalk": "^4.0.0", + "cjs-module-lexer": "^1.0.0", + "collect-v8-coverage": "^1.0.0", + "execa": "^5.0.0", + "glob": "^7.1.3", + "graceful-fs": "^4.2.9", + "jest-haste-map": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-mock": "^27.5.1", + "jest-regex-util": "^27.5.1", + "jest-resolve": "^27.5.1", + "jest-snapshot": "^27.5.1", + "jest-util": "^27.5.1", + "slash": "^3.0.0", + "strip-bom": "^4.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-serializer": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-27.5.1.tgz", + "integrity": "sha512-jZCyo6iIxO1aqUxpuBlwTDMkzOAJS4a3eYz3YzgxxVQFwLeSA7Jfq5cbqCY+JLvTDrWirgusI/0KwxKMgrdf7w==", + "dev": true, + "requires": { + "@types/node": "*", + "graceful-fs": "^4.2.9" + } + }, + "jest-snapshot": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-27.5.1.tgz", + "integrity": "sha512-yYykXI5a0I31xX67mgeLw1DZ0bJB+gpq5IpSuCAoyDi0+BhgU/RIrL+RTzDmkNTchvDFWKP8lp+w/42Z3us5sA==", + "dev": true, + "requires": { + "@babel/core": "^7.7.2", + "@babel/generator": "^7.7.2", + "@babel/plugin-syntax-typescript": "^7.7.2", + "@babel/traverse": "^7.7.2", + "@babel/types": "^7.0.0", + "@jest/transform": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/babel__traverse": "^7.0.4", + "@types/prettier": "^2.1.5", + "babel-preset-current-node-syntax": "^1.0.0", + "chalk": "^4.0.0", + "expect": "^27.5.1", + "graceful-fs": "^4.2.9", + "jest-diff": "^27.5.1", + "jest-get-type": "^27.5.1", + "jest-haste-map": "^27.5.1", + "jest-matcher-utils": "^27.5.1", + "jest-message-util": "^27.5.1", + "jest-util": "^27.5.1", + "natural-compare": "^1.4.0", + "pretty-format": "^27.5.1", + "semver": "^7.3.2" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-util": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-27.5.1.tgz", + "integrity": "sha512-Kv2o/8jNvX1MQ0KGtw480E/w4fBCDOnH6+6DmeKi6LZUIlKA5kwY0YNdlzaWTiVgxqAqik11QyxDOKk543aKXw==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "@types/node": "*", + "chalk": "^4.0.0", + "ci-info": "^3.2.0", + "graceful-fs": "^4.2.9", + "picomatch": "^2.2.3" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-validate": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-27.5.1.tgz", + "integrity": "sha512-thkNli0LYTmOI1tDB3FI1S1RTp/Bqyd9pTarJwL87OIBFuqEb5Apv5EaApEudYg4g86e3CT6kM0RowkhtEnCBQ==", + "dev": true, + "requires": { + "@jest/types": "^27.5.1", + "camelcase": "^6.2.0", + "chalk": "^4.0.0", + "jest-get-type": "^27.5.1", + "leven": "^3.1.0", + "pretty-format": "^27.5.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "camelcase": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", + "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", + "dev": true + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-watcher": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-27.5.1.tgz", + "integrity": "sha512-z676SuD6Z8o8qbmEGhoEUFOM1+jfEiL3DXHK/xgEiG2EyNYfFG60jluWcupY6dATjfEsKQuibReS1djInQnoVw==", + "dev": true, + "requires": { + "@jest/test-result": "^27.5.1", + "@jest/types": "^27.5.1", + "@types/node": "*", + "ansi-escapes": "^4.2.1", + "chalk": "^4.0.0", + "jest-util": "^27.5.1", + "string-length": "^4.0.1" + }, + "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "jest-worker": { + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.5.1.tgz", + "integrity": "sha512-7vuh85V5cdDofPyxn58nrPjBktZo0u9x1g8WtjQol+jZDaE+fhN+cIvTj11GndBnMnyfrUOG1sZQxCdjKh+DKg==", + "dev": true, + "requires": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^8.0.0" + }, + "dependencies": { + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "joi": { + "version": "17.6.0", + "resolved": "https://registry.npmjs.org/joi/-/joi-17.6.0.tgz", + "integrity": "sha512-OX5dG6DTbcr/kbMFj0KGYxuew69HPcAE3K/sZpEV2nP6e/j/C0HV+HNiBPCASxdx5T7DMoa0s8UeHWMnb6n2zw==", + "dev": true, + "requires": { + "@hapi/hoek": "^9.0.0", + "@hapi/topo": "^5.0.0", + "@sideway/address": "^4.1.3", + "@sideway/formula": "^3.0.0", + "@sideway/pinpoint": "^2.0.0" + } + }, + "js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", + "dev": true + }, + "js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "requires": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + } + }, + "js2xmlparser": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/js2xmlparser/-/js2xmlparser-4.0.2.tgz", + "integrity": "sha512-6n4D8gLlLf1n5mNLQPRfViYzu9RATblzPEtm1SthMX1Pjao0r9YI9nw7ZIfRxQMERS87mcswrg+r/OYrPRX6jA==", + "dev": true, + "requires": { + "xmlcreate": "^2.0.4" + } + }, + "jsbn": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", + "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", "dev": true }, - "fancy-test": { - "version": "1.4.10", - "resolved": "https://registry.npmjs.org/fancy-test/-/fancy-test-1.4.10.tgz", - "integrity": "sha512-AaUX6wKS7D5OP2YK2q5G7c8PGx2lgoyLUD7Bbg8z323sb9aebBqzb9UN6phzI73UgO/ViihmNfOxF3kdfZLhew==", + "jsdoc": { + "version": "3.6.10", + "resolved": "https://registry.npmjs.org/jsdoc/-/jsdoc-3.6.10.tgz", + "integrity": "sha512-IdQ8ppSo5LKZ9o3M+LKIIK8i00DIe5msDvG3G81Km+1dhy0XrOWD0Ji8H61ElgyEj/O9KRLokgKbAM9XX9CJAg==", + "dev": true, + "requires": { + "@babel/parser": "^7.9.4", + "@types/markdown-it": "^12.2.3", + "bluebird": "^3.7.2", + "catharsis": "^0.9.0", + "escape-string-regexp": "^2.0.0", + "js2xmlparser": "^4.0.2", + "klaw": "^4.0.1", + "markdown-it": "^12.3.2", + "markdown-it-anchor": "^8.4.1", + "marked": "^4.0.10", + "mkdirp": "^1.0.4", + "requizzle": "^0.2.3", + "strip-json-comments": "^3.1.0", + "taffydb": "2.6.2", + "underscore": "~1.13.2" + }, + "dependencies": { + "escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "dev": true + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + } + } + }, + "jsdoc-api": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/jsdoc-api/-/jsdoc-api-7.1.1.tgz", + "integrity": "sha512-0pkuPCzVXiqsDAsVrNFXCkHzlyNepBIDVtwwehry4RJAnZmXtlAz7rh8F9FRz53u3NeynGbex+bpYWwi8lE66A==", "dev": true, "requires": { - "@types/chai": "*", - "@types/lodash": "*", - "@types/node": "*", - "@types/sinon": "*", - "lodash": "^4.17.13", - "mock-stdin": "^1.0.0", - "nock": "^13.0.0", - "stdout-stderr": "^0.1.9" + "array-back": "^6.2.2", + "cache-point": "^2.0.0", + "collect-all": "^1.0.4", + "file-set": "^4.0.2", + "fs-then-native": "^2.0.0", + "jsdoc": "^3.6.10", + "object-to-spawn-args": "^2.0.1", + "temp-path": "^1.0.0", + "walk-back": "^5.1.0" } }, - "fast-deep-equal": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", - "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "jsdoc-parse": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsdoc-parse/-/jsdoc-parse-6.1.0.tgz", + "integrity": "sha512-n/hDGQJa69IBun1yZAjqzV4gVR41+flZ3bIlm9fKvNe2Xjsd1/+zCo2+R9ls8LxtePgIWbpA1jU7xkB2lRdLLg==", + "dev": true, + "requires": { + "array-back": "^6.2.2", + "lodash.omit": "^4.5.0", + "lodash.pick": "^4.4.0", + "reduce-extract": "^1.0.0", + "sort-array": "^4.1.4", + "test-value": "^3.0.0" + } + }, + "jsdoc-to-markdown": { + "version": "7.1.1", + "resolved": "https://registry.npmjs.org/jsdoc-to-markdown/-/jsdoc-to-markdown-7.1.1.tgz", + "integrity": "sha512-CI86d63xAVNO+ENumWwmJ034lYe5iGU5GwjtTA11EuphP9tpnoi4hrKgR/J8uME0D+o4KUpVfwX1fjZhc8dEtg==", + "dev": true, + "requires": { + "array-back": "^6.2.2", + "command-line-tool": "^0.8.0", + "config-master": "^3.1.0", + "dmd": "^6.1.0", + "jsdoc-api": "^7.1.1", + "jsdoc-parse": "^6.1.0", + "walk-back": "^5.1.0" + } + }, + "jsdom": { + "version": "16.7.0", + "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-16.7.0.tgz", + "integrity": "sha512-u9Smc2G1USStM+s/x1ru5Sxrl6mPYCbByG1U/hUmqaVsm4tbNyS7CicOSRyuGQYZhTu0h84qkZZQ/I+dzizSVw==", + "dev": true, + "requires": { + "abab": "^2.0.5", + "acorn": "^8.2.4", + "acorn-globals": "^6.0.0", + "cssom": "^0.4.4", + "cssstyle": "^2.3.0", + "data-urls": "^2.0.0", + "decimal.js": "^10.2.1", + "domexception": "^2.0.1", + "escodegen": "^2.0.0", + "form-data": "^3.0.0", + "html-encoding-sniffer": "^2.0.1", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-potential-custom-element-name": "^1.0.1", + "nwsapi": "^2.2.0", + "parse5": "6.0.1", + "saxes": "^5.0.1", + "symbol-tree": "^3.2.4", + "tough-cookie": "^4.0.0", + "w3c-hr-time": "^1.0.2", + "w3c-xmlserializer": "^2.0.0", + "webidl-conversions": "^6.1.0", + "whatwg-encoding": "^1.0.5", + "whatwg-mimetype": "^2.3.0", + "whatwg-url": "^8.5.0", + "ws": "^7.4.6", + "xml-name-validator": "^3.0.0" + }, + "dependencies": { + "form-data": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", + "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", + "dev": true, + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + } + } + } + }, + "jsen": { + "version": "0.6.6", + "resolved": "https://registry.npmjs.org/jsen/-/jsen-0.6.6.tgz", + "integrity": "sha1-AkDBjPETUKwCFFb0in6xO9Z+BCA=", "dev": true }, - "fast-glob": { - "version": "3.2.7", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.7.tgz", - "integrity": "sha512-rYGMRwip6lUMvYD3BTScMwT1HtAs2d71SMv66Vrxs0IekGZEjhM0pcMfjQPnknBt2zeCwQMEupiN02ZP4DiT1Q==", + "jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "dev": true + }, + "jsforce": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/jsforce/-/jsforce-1.11.0.tgz", + "integrity": "sha512-vYNXJXXdz9ZQNdfRqq/MCJ/zU7JGA7iEduwafQDzChR9FeqXgTNfHTppLVbw9mIniKkQZemmxSOtl7N04lj/5Q==", "dev": true, "requires": { - "@nodelib/fs.stat": "^2.0.2", - "@nodelib/fs.walk": "^1.2.3", - "glob-parent": "^5.1.2", - "merge2": "^1.3.0", - "micromatch": "^4.0.4" + "base64-url": "^2.2.0", + "co-prompt": "^1.0.0", + "coffeescript": "^1.10.0", + "commander": "^2.9.0", + "csv-parse": "^4.10.1", + "csv-stringify": "^1.0.4", + "faye": "^1.4.0", + "inherits": "^2.0.1", + "lodash": "^4.17.19", + "multistream": "^2.0.5", + "opn": "^5.3.0", + "promise": "^7.1.1", + "readable-stream": "^2.1.0", + "request": "^2.72.0", + "xml2js": "^0.4.16" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } } }, - "fast-json-stable-stringify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", - "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", + "json-buffer": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/json-buffer/-/json-buffer-3.0.1.tgz", + "integrity": "sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ==", "dev": true }, - "fast-levenshtein": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", - "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", + "json-parse-better-errors": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", + "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", "dev": true }, - "fastq": { - "version": "1.13.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.13.0.tgz", - "integrity": "sha512-YpkpUnK8od0o1hmeSc7UUs/eB/vIPWJYjKck2QKIzAf71Vm1AAQ3EbuZB3g2JIy+pg+ERD0vqI79KyZiB2e2Nw==", + "json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true + }, + "json-schema": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz", + "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==", + "dev": true + }, + "json-schema-traverse": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", + "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", + "dev": true + }, + "json-stable-stringify-without-jsonify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", + "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", + "dev": true + }, + "json-stringify-safe": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", + "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", + "dev": true + }, + "json5": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.1.tgz", + "integrity": "sha512-1hqLFMSrGHRHxav9q9gNjJ5EXznIxGVO09xQRrwplcS8qs28pZ8s8hupZAmqDwZUmVZ2Qb2jnyPOWcDH8m8dlA==", + "dev": true + }, + "jsonfile": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", + "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", "dev": true, "requires": { - "reusify": "^1.0.4" + "graceful-fs": "^4.1.6" } }, - "faye": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/faye/-/faye-1.4.0.tgz", - "integrity": "sha512-kRrIg4be8VNYhycS2PY//hpBJSzZPr/DBbcy9VWelhZMW3KhyLkQR0HL0k0MNpmVoNFF4EdfMFkNAWjTP65g6w==", + "jsonparse": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", + "integrity": "sha1-P02uSpH6wxX3EGL4UhzCOfE2YoA=", + "dev": true + }, + "jsonwebtoken": { + "version": "8.5.0", + "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.0.tgz", + "integrity": "sha512-IqEycp0znWHNA11TpYi77bVgyBO/pGESDh7Ajhas+u0ttkGkKYIIAjniL4Bw5+oVejVF+SYkaI7XKfwCCyeTuA==", "dev": true, "requires": { - "asap": "*", - "csprng": "*", - "faye-websocket": ">=0.9.1", - "safe-buffer": "*", - "tough-cookie": "*", - "tunnel-agent": "*" + "jws": "^3.2.1", + "lodash.includes": "^4.3.0", + "lodash.isboolean": "^3.0.3", + "lodash.isinteger": "^4.0.4", + "lodash.isnumber": "^3.0.3", + "lodash.isplainobject": "^4.0.6", + "lodash.isstring": "^4.0.1", + "lodash.once": "^4.0.0", + "ms": "^2.1.1", + "semver": "^5.6.0" + }, + "dependencies": { + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true + } } }, - "faye-websocket": { - "version": "0.11.4", - "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.4.tgz", - "integrity": "sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==", + "jsprim": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.2.tgz", + "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==", "dev": true, "requires": { - "websocket-driver": ">=0.5.1" + "assert-plus": "1.0.0", + "extsprintf": "1.3.0", + "json-schema": "0.4.0", + "verror": "1.10.0" } }, - "fb-watchman": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/fb-watchman/-/fb-watchman-2.0.1.tgz", - "integrity": "sha512-DkPJKQeY6kKwmuMretBhr7G6Vodr7bFwDYTXIkfG1gjvNpaxBTQV3PbXg6bR1c1UP4jPOX0jHUbbHANL9vRjVg==", + "jwa": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", + "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", + "dev": true, + "requires": { + "buffer-equal-constant-time": "1.0.1", + "ecdsa-sig-formatter": "1.0.11", + "safe-buffer": "^5.0.1" + } + }, + "jws": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", + "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", "dev": true, "requires": { - "bser": "2.1.1" + "jwa": "^1.4.1", + "safe-buffer": "^5.0.1" } }, - "figures": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", - "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "keypress": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/keypress/-/keypress-0.2.1.tgz", + "integrity": "sha1-HoBFQlABjbrUw/6USX1uZ7YmnHc=", + "dev": true + }, + "keyv": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/keyv/-/keyv-4.2.2.tgz", + "integrity": "sha512-uYS0vKTlBIjNCAUqrjlxmruxOEiZxZIHXyp32sdcGmP+ukFrmWUnE//RcPXJH3Vxrni1H2gsQbjHE0bH7MtMQQ==", "dev": true, "requires": { - "escape-string-regexp": "^1.0.5" + "compress-brotli": "^1.3.6", + "json-buffer": "3.0.1" } }, - "file-entry-cache": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/file-entry-cache/-/file-entry-cache-5.0.1.tgz", - "integrity": "sha512-bCg29ictuBaKUwwArK4ouCaqDgLZcysCFLmM/Yn/FDoqndh/9vNuQfXRDvTuXKLxfD/JtZQGKFT8MGcJBK644g==", + "kind-of": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", + "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", + "dev": true + }, + "klaw": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/klaw/-/klaw-4.0.1.tgz", + "integrity": "sha512-pgsE40/SvC7st04AHiISNewaIMUbY5V/K8b21ekiPiFoYs/EYSdsGa+FJArB1d441uq4Q8zZyIxvAzkGNlBdRw==", + "dev": true + }, + "kleur": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", + "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", + "dev": true + }, + "lazy-cache": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/lazy-cache/-/lazy-cache-2.0.2.tgz", + "integrity": "sha1-uRkKT5EzVGlIQIWfio9whNiCImQ=", "dev": true, "requires": { - "flat-cache": "^2.0.1" + "set-getter": "^0.1.0" } }, - "file-set": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/file-set/-/file-set-4.0.2.tgz", - "integrity": "sha512-fuxEgzk4L8waGXaAkd8cMr73Pm0FxOVkn8hztzUW7BAHhOGH90viQNXbiOsnecCWmfInqU6YmAMwxRMdKETceQ==", + "lazystream": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lazystream/-/lazystream-1.0.1.tgz", + "integrity": "sha512-b94GiNHQNy6JNTrt5w6zNyffMrNkXZb3KTkCZJb2V1xaEGCk093vkZ2jk3tpaeP33/OiXC+WvK9AxUebnf5nbw==", "dev": true, "requires": { - "array-back": "^5.0.0", - "glob": "^7.1.6" + "readable-stream": "^2.0.5" }, "dependencies": { - "array-back": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-5.0.0.tgz", - "integrity": "sha512-kgVWwJReZWmVuWOQKEOohXKJX+nD02JAZ54D1RRWlv8L0NebauKAaFxACKzB74RTclt1+WNz5KHaLRDAPZbDEw==", + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", "dev": true } } }, - "fill-range": { - "version": "7.0.1", - "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", - "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "leven": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", + "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", + "dev": true + }, + "levn": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", + "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", "dev": true, "requires": { - "to-regex-range": "^5.0.1" + "prelude-ls": "~1.1.2", + "type-check": "~0.3.2" } }, - "find-java-home": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/find-java-home/-/find-java-home-1.1.0.tgz", - "integrity": "sha512-bSTCKNZ193UM/+ZZoNDzICAEHcVywovkhsWCkZALjCvRXQ+zXTe/XATrrP4CpxkaP6YFhQJOpyRpH0P2U/woDA==", + "lilconfig": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.0.4.tgz", + "integrity": "sha512-bfTIN7lEsiooCocSISTWXkiWJkRqtL9wYtYy+8EK3Y41qh3mpwPU0ycTOgjdY9ErwXCc8QyrQp82bdL0Xkm9yA==", + "dev": true + }, + "line-column": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/line-column/-/line-column-1.0.2.tgz", + "integrity": "sha1-0lryk2tvSEkXKzEuR5LR2Ye8NKI=", "dev": true, "requires": { - "which": "~1.0.5", - "winreg": "~1.2.2" - }, - "dependencies": { - "which": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/which/-/which-1.0.9.tgz", - "integrity": "sha1-RgwdoPgQED0DIam2M6+eV15kSG8=", - "dev": true - } + "isarray": "^1.0.0", + "isobject": "^2.0.0" } }, - "find-replace": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/find-replace/-/find-replace-3.0.0.tgz", - "integrity": "sha512-6Tb2myMioCAgv5kfvP5/PkZZ/ntTpVK39fHY7WkWBgvbeE+VHd/tZuZ4mrC+bxh4cfOZeYKVPaJIZtZXV7GNCQ==", + "lines-and-columns": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", + "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", + "dev": true + }, + "lint-staged": { + "version": "12.3.7", + "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-12.3.7.tgz", + "integrity": "sha512-/S4D726e2GIsDVWIk1XGvheCaDm1SJRQp8efamZFWJxQMVEbOwSysp7xb49Oo73KYCdy97mIWinhlxcoNqIfIQ==", "dev": true, "requires": { - "array-back": "^3.0.1" + "cli-truncate": "^3.1.0", + "colorette": "^2.0.16", + "commander": "^8.3.0", + "debug": "^4.3.3", + "execa": "^5.1.1", + "lilconfig": "2.0.4", + "listr2": "^4.0.1", + "micromatch": "^4.0.4", + "normalize-path": "^3.0.0", + "object-inspect": "^1.12.0", + "pidtree": "^0.5.0", + "string-argv": "^0.3.1", + "supports-color": "^9.2.1", + "yaml": "^1.10.2" }, "dependencies": { - "array-back": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-3.1.0.tgz", - "integrity": "sha512-TkuxA4UCOvxuDK6NZYXCalszEzj+TLszyASooky+i742l9TqsOdYCMJJupxRic61hwquNtppB3hgcuq9SVSH1Q==", + "commander": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", + "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", + "dev": true + }, + "supports-color": { + "version": "9.2.2", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-9.2.2.tgz", + "integrity": "sha512-XC6g/Kgux+rJXmwokjm9ECpD6k/smUoS5LKlUCcsYr4IY3rW0XyAympon2RmxGrlnZURMpg5T18gWDP9CsHXFA==", "dev": true } } }, - "find-up": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", - "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", - "dev": true, - "requires": { - "locate-path": "^6.0.0", - "path-exists": "^4.0.0" - } + "listenercount": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/listenercount/-/listenercount-1.0.1.tgz", + "integrity": "sha1-hMinKrWcRyUyFIDJdeZQg0LnCTc=", + "dev": true }, - "flat-cache": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-2.0.1.tgz", - "integrity": "sha512-LoQe6yDuUMDzQAEH8sgmh4Md6oZnc/7PjtwjNFSzveXqSHt6ka9fPBuso7IGf9Rz4uqnSnWiFH2B/zj24a5ReA==", + "listr2": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/listr2/-/listr2-4.0.5.tgz", + "integrity": "sha512-juGHV1doQdpNT3GSTs9IUN43QJb7KHdF9uqg7Vufs/tG9VTzpFphqF4pm/ICdAABGQxsyNn9CiYA3StkI6jpwA==", "dev": true, "requires": { - "flatted": "^2.0.0", - "rimraf": "2.6.3", - "write": "1.0.3" + "cli-truncate": "^2.1.0", + "colorette": "^2.0.16", + "log-update": "^4.0.0", + "p-map": "^4.0.0", + "rfdc": "^1.3.0", + "rxjs": "^7.5.5", + "through": "^2.3.8", + "wrap-ansi": "^7.0.0" }, "dependencies": { - "rimraf": { - "version": "2.6.3", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.6.3.tgz", - "integrity": "sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==", + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", "dev": true, "requires": { - "glob": "^7.1.3" + "color-convert": "^2.0.1" + } + }, + "astral-regex": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", + "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", + "dev": true + }, + "cli-truncate": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/cli-truncate/-/cli-truncate-2.1.0.tgz", + "integrity": "sha512-n8fOixwDD6b/ObinzTrp1ZKFzbgvKZvuz/TvejnLn1aQfC6r52XEx85FmuC+3HI+JM7coBRXUvNqEU2PHVrHpg==", + "dev": true, + "requires": { + "slice-ansi": "^3.0.0", + "string-width": "^4.2.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "rxjs": { + "version": "7.5.5", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.5.5.tgz", + "integrity": "sha512-sy+H0pQofO95VDmFLzyaw9xNJU4KTRSwQIGM6+iG3SypAtCiLDzpeG8sJrNCWn2Up9km+KhkvTdbkrdy+yzZdw==", + "dev": true, + "requires": { + "tslib": "^2.1.0" + } + }, + "slice-ansi": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-3.0.0.tgz", + "integrity": "sha512-pSyv7bSTC7ig9Dcgbw9AuRNUb5k5V6oDudjZoMBSr13qpLBG7tB+zgCkARjq7xIUgdz5P1Qe8u+rSGdouOOIyQ==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "astral-regex": "^2.0.0", + "is-fullwidth-code-point": "^3.0.0" } } } }, - "flatted": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", - "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", - "dev": true - }, - "follow-redirects": { - "version": "1.14.8", - "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.14.8.tgz", - "integrity": "sha512-1x0S9UVJHsQprFcEC/qnNzBLcIxsjAV905f/UkQxbclCsoTWlacCNOpQa/anodLl2uaEKFhfWOvM2Qg77+15zA==" - }, - "forever-agent": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", - "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", - "dev": true - }, - "form-data": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", - "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", - "dev": true, - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.8", - "mime-types": "^2.1.12" - } - }, - "fs-constants": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz", - "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==", - "dev": true - }, - "fs-extra": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", - "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "load-json-file": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz", + "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg=", "dev": true, "requires": { - "graceful-fs": "^4.2.0", - "jsonfile": "^4.0.0", - "universalify": "^0.1.0" + "graceful-fs": "^4.1.2", + "parse-json": "^2.2.0", + "pify": "^2.0.0", + "strip-bom": "^3.0.0" + }, + "dependencies": { + "parse-json": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", + "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", + "dev": true, + "requires": { + "error-ex": "^1.2.0" + } + }, + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + } } }, - "fs-minipass": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz", - "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", + "locate-path": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", "dev": true, "requires": { - "minipass": "^2.6.0" + "p-locate": "^5.0.0" } }, - "fs-then-native": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/fs-then-native/-/fs-then-native-2.0.0.tgz", - "integrity": "sha1-GaEk2U2QwiyOBF8ujdbr6jbUjGc=", + "lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", "dev": true }, - "fs.realpath": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", - "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" - }, - "fsevents": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", - "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", - "dev": true, - "optional": true - }, - "function-bind": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" - }, - "functional-red-black-tree": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/functional-red-black-tree/-/functional-red-black-tree-1.0.1.tgz", - "integrity": "sha1-GwqzvVU7Kg1jmdKcDj6gslIHgyc=", + "lodash._reinterpolate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz", + "integrity": "sha1-DM8tiRZq8Ds2Y8eWU4t1rG4RTZ0=", "dev": true }, - "gensync": { - "version": "1.0.0-beta.2", - "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.2.tgz", - "integrity": "sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==", + "lodash.camelcase": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz", + "integrity": "sha1-soqmKIorn8ZRA1x3EfZathkDMaY=", "dev": true }, - "get-caller-file": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", - "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "lodash.defaults": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", + "integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw=", "dev": true }, - "get-intrinsic": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", - "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", - "dev": true, - "requires": { - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1" - } - }, - "get-package-type": { - "version": "0.1.0", - "resolved": "https://registry.npmjs.org/get-package-type/-/get-package-type-0.1.0.tgz", - "integrity": "sha512-pjzuKtY64GYfWizNAJ0fr9VqttZkNiK2iS430LtIHzjBEr6bX8Am2zm4sW4Ro5wjWW5cAlRL1qAMTcXbjNAO2Q==", + "lodash.difference": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", + "integrity": "sha1-nMtOUF1Ia5FlE0V3KIWi3yf9AXw=", "dev": true }, - "get-stream": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-6.0.1.tgz", - "integrity": "sha512-ts6Wi+2j3jQjqi70w5AlN8DFnkSwC+MqmxEzdEALB2qXZYV3X/b1CTfgPLGJNMeAWxdPfU8FO1ms3NUfaHCPYg==", + "lodash.flatten": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", + "integrity": "sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8=", "dev": true }, - "get-symbol-description": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz", - "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.1" - } + "lodash.get": { + "version": "4.4.2", + "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz", + "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=", + "dev": true }, - "getpass": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", - "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", - "dev": true, - "requires": { - "assert-plus": "^1.0.0" - } + "lodash.includes": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", + "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=", + "dev": true }, - "glob": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", - "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", - "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" - } + "lodash.isboolean": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", + "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=", + "dev": true }, - "glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", - "dev": true, - "requires": { - "is-glob": "^4.0.1" - } + "lodash.isinteger": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", + "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=", + "dev": true }, - "globals": { - "version": "11.12.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", - "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", + "lodash.isnumber": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", + "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=", "dev": true }, - "globby": { - "version": "11.0.4", - "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.4.tgz", - "integrity": "sha512-9O4MVG9ioZJ08ffbcyVYyLOJLk5JQ688pJ4eMGLpdWLHq/Wr1D9BlriLQyL0E+jbkuePVZXYFj47QM/v093wHg==", - "dev": true, - "requires": { - "array-union": "^2.1.0", - "dir-glob": "^3.0.1", - "fast-glob": "^3.1.1", - "ignore": "^5.1.4", - "merge2": "^1.3.0", - "slash": "^3.0.0" - } + "lodash.isplainobject": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", + "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=", + "dev": true }, - "graceful-fs": { - "version": "4.2.8", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", - "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", + "lodash.isstring": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", + "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=", "dev": true }, - "handlebars": { - "version": "4.7.7", - "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.7.tgz", - "integrity": "sha512-aAcXm5OAfE/8IXkcZvCepKU3VzW1/39Fb5ZuqMtgI/hT8X2YgoMvBY5dLhq/cpOvw7Lk1nK/UF71aLG/ZnVYRA==", - "dev": true, - "requires": { - "minimist": "^1.2.5", - "neo-async": "^2.6.0", - "source-map": "^0.6.1", - "uglify-js": "^3.1.4", - "wordwrap": "^1.0.0" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } - } + "lodash.memoize": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", + "integrity": "sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4=", + "dev": true }, - "har-schema": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", - "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", + "lodash.merge": { + "version": "4.6.2", + "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", + "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", "dev": true }, - "har-validator": { - "version": "5.1.5", - "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", - "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", - "dev": true, - "requires": { - "ajv": "^6.12.3", - "har-schema": "^2.0.0" - } + "lodash.omit": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.omit/-/lodash.omit-4.5.0.tgz", + "integrity": "sha1-brGa5aHuHdnfC5aeZs4Lf6MLXmA=", + "dev": true }, - "has": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", - "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", - "requires": { - "function-bind": "^1.1.1" - } + "lodash.once": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", + "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=", + "dev": true }, - "has-bigints": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.1.tgz", - "integrity": "sha512-LSBS2LjbNBTf6287JEbEzvJgftkF5qFkmCo9hDRpAzKhUOlJ+hx8dd4USs00SgsUNwc4617J9ki5YtEClM2ffA==", + "lodash.padend": { + "version": "4.6.1", + "resolved": "https://registry.npmjs.org/lodash.padend/-/lodash.padend-4.6.1.tgz", + "integrity": "sha1-U8y6BH0G4VjTEfRdpiX05J5vFm4=", "dev": true }, - "has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", + "lodash.pick": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/lodash.pick/-/lodash.pick-4.4.0.tgz", + "integrity": "sha1-UvBWEP/53tQiYRRB7R/BI6AwAbM=", "dev": true }, - "has-symbols": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.2.tgz", - "integrity": "sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw==", + "lodash.set": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/lodash.set/-/lodash.set-4.3.2.tgz", + "integrity": "sha1-2HV7HagH3eJIFrDWqEvqGnYjCyM=", "dev": true }, - "has-tostringtag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz", - "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", + "lodash.template": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/lodash.template/-/lodash.template-4.5.0.tgz", + "integrity": "sha512-84vYFxIkmidUiFxidA/KjjH9pAycqW+h980j7Fuz5qxRtO9pgB7MDFTdys1N7A5mcucRiDyEq4fusljItR1T/A==", "dev": true, "requires": { - "has-symbols": "^1.0.2" + "lodash._reinterpolate": "^3.0.0", + "lodash.templatesettings": "^4.0.0" } }, - "he": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/he/-/he-1.2.0.tgz", - "integrity": "sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==", - "dev": true - }, - "hosted-git-info": { - "version": "2.8.9", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz", - "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", - "dev": true - }, - "html-encoding-sniffer": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-2.0.1.tgz", - "integrity": "sha512-D5JbOMBIR/TVZkubHT+OyT2705QvogUW4IBn6nHd756OwieSF9aDYFj4dv6HHEVGYbHaLETa3WggZYWWMyy3ZQ==", + "lodash.templatesettings": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/lodash.templatesettings/-/lodash.templatesettings-4.2.0.tgz", + "integrity": "sha512-stgLz+i3Aa9mZgnjr/O+v9ruKZsPsndy7qPZOchbqk2cnTU1ZaldKK+v7m54WoKIyxiuMZTKT2H81F8BeAc3ZQ==", "dev": true, "requires": { - "whatwg-encoding": "^1.0.5" + "lodash._reinterpolate": "^3.0.0" } }, - "html-entities": { - "version": "2.3.2", - "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-2.3.2.tgz", - "integrity": "sha512-c3Ab/url5ksaT0WyleslpBEthOzWhrjQbg75y7XUsfSzi3Dgzt0l8w5e7DylRn15MTlMMD58dTfzddNS2kcAjQ==", - "dev": true - }, - "html-escaper": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", - "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", - "dev": true - }, - "http-parser-js": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.5.tgz", - "integrity": "sha512-x+JVEkO2PoM8qqpbPbOL3cqHPwerep7OwzK7Ay+sMQjKzaKCqWvjoXm5tqMP9tXWWTnTzAjIhXg+J99XYuPhPA==", + "lodash.union": { + "version": "4.6.0", + "resolved": "https://registry.npmjs.org/lodash.union/-/lodash.union-4.6.0.tgz", + "integrity": "sha1-SLtQiECfFvGCFmZkHETdGqrjzYg=", "dev": true }, - "http-proxy-agent": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", - "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", + "log-chopper": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/log-chopper/-/log-chopper-1.0.2.tgz", + "integrity": "sha512-tEWS6Fb+Xv0yLChJ6saA1DP3H1yPL0PfiIN7SDJ+U/CyP+fD4G/dhKfow+P5UuJWi6BdE4mUcPkJclGXCWxDrg==", "dev": true, "requires": { - "@tootallnate/once": "1", - "agent-base": "6", - "debug": "4" + "byline": "5.x" } }, - "http-signature": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", - "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", + "log-symbols": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", + "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", "dev": true, "requires": { - "assert-plus": "^1.0.0", - "jsprim": "^1.2.2", - "sshpk": "^1.7.0" + "chalk": "^2.0.1" } }, - "https-proxy-agent": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", - "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", + "log-update": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/log-update/-/log-update-4.0.0.tgz", + "integrity": "sha512-9fkkDevMefjg0mmzWFBW8YkFP91OrizzkW3diF7CpG+S2EYdy4+TVfGwz1zeF8x7hCx1ovSPTOE9Ngib74qqUg==", "dev": true, "requires": { - "agent-base": "6", - "debug": "4" + "ansi-escapes": "^4.3.0", + "cli-cursor": "^3.1.0", + "slice-ansi": "^4.0.0", + "wrap-ansi": "^6.2.0" + }, + "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "astral-regex": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", + "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", + "dev": true + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "slice-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz", + "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "astral-regex": "^2.0.0", + "is-fullwidth-code-point": "^3.0.0" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + } + } } }, - "human-signals": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-2.1.0.tgz", - "integrity": "sha512-B4FFZ6q/T2jhhksgkbEW3HBvWIfDW85snkQgawt07S7J5QXTk6BkNV+0yAeZrM5QpMAdYlocGoljn0sJ/WQkFw==", - "dev": true - }, - "husky": { - "version": "7.0.4", - "resolved": "https://registry.npmjs.org/husky/-/husky-7.0.4.tgz", - "integrity": "sha512-vbaCKN2QLtP/vD4yvs6iz6hBEo6wkSzs8HpRah1Z6aGmF2KW5PdYuAd7uX5a+OyBZHBhd+TFLqgjUgytQr4RvQ==", - "dev": true - }, - "hyperlinker": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/hyperlinker/-/hyperlinker-1.0.0.tgz", - "integrity": "sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ==", - "dev": true - }, - "iconv-lite": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", - "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "lower-case": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/lower-case/-/lower-case-2.0.2.tgz", + "integrity": "sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==", "dev": true, "requires": { - "safer-buffer": ">= 2.1.2 < 3" + "tslib": "^2.0.3" } }, - "ieee754": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz", - "integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==", - "dev": true - }, - "ignore": { - "version": "5.1.9", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.9.tgz", - "integrity": "sha512-2zeMQpbKz5dhZ9IwL0gbxSW5w0NK/MSAMtNuhgIHEPmaU3vPdKPL0UdvUCXs5SS4JAwsBxysK5sFMW8ocFiVjQ==", + "lowercase-keys": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-1.0.1.tgz", + "integrity": "sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA==", "dev": true }, - "import-fresh": { - "version": "3.3.0", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", - "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", "dev": true, "requires": { - "parent-module": "^1.0.0", - "resolve-from": "^4.0.0" + "yallist": "^4.0.0" }, "dependencies": { - "resolve-from": { + "yallist": { "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", "dev": true } } }, - "import-local": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.0.3.tgz", - "integrity": "sha512-bE9iaUY3CXH8Cwfan/abDKAxe1KGT9kyGsBPqf6DMK/z0a2OzAsrukeYNgIH6cH5Xr452jb1TUL8rSfCLjZ9uA==", + "make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", "dev": true, "requires": { - "pkg-dir": "^4.2.0", - "resolve-cwd": "^3.0.0" + "semver": "^6.0.0" } }, - "imurmurhash": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", - "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true + }, + "makeerror": { + "version": "1.0.12", + "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", + "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", + "dev": true, + "requires": { + "tmpl": "1.0.5" + } + }, + "map-cache": { + "version": "0.2.2", + "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", + "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", "dev": true }, - "indent-string": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", - "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", + "map-visit": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", + "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", + "dev": true, + "requires": { + "object-visit": "^1.0.0" + } + }, + "markdown-it": { + "version": "12.3.2", + "resolved": "https://registry.npmjs.org/markdown-it/-/markdown-it-12.3.2.tgz", + "integrity": "sha512-TchMembfxfNVpHkbtriWltGWc+m3xszaRD0CZup7GFFhzIgQqxIfn3eGj1yZpfuflzPvfkt611B2Q/Bsk1YnGg==", + "requires": { + "argparse": "^2.0.1", + "entities": "~2.1.0", + "linkify-it": "^3.0.1", + "mdurl": "^1.0.1", + "uc.micro": "^1.0.5" + }, + "dependencies": { + "argparse": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" + }, + "entities": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.1.0.tgz", + "integrity": "sha512-hCx1oky9PFrJ611mf0ifBLBRW8lUUVRlFolb5gWRfIELabBlbp9xZvrqZLZAs+NxFnbfQoeGd8wDkygjg7U85w==" + }, + "linkify-it": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/linkify-it/-/linkify-it-3.0.3.tgz", + "integrity": "sha512-ynTsyrFSdE5oZ/O9GEf00kPngmOfVwazR5GKDq6EYfhlpFug3J2zybX56a2PRRpc9P+FuSoGNAwjlbDs9jJBPQ==", + "requires": { + "uc.micro": "^1.0.1" + } + } + } + }, + "markdown-it-anchor": { + "version": "8.6.2", + "resolved": "https://registry.npmjs.org/markdown-it-anchor/-/markdown-it-anchor-8.6.2.tgz", + "integrity": "sha512-JNaekTlIwwyYGBN3zifZDxgz4bSL8sbEj58fdTZGmPSMMGXBZapFjcZk2I33Jy79c1fvCKHpF7MA/67FOTjvzA==", "dev": true }, - "inflight": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", - "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", - "requires": { - "once": "^1.3.0", - "wrappy": "1" - } - }, - "inherits": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", - "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + "marked": { + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/marked/-/marked-4.0.10.tgz", + "integrity": "sha512-+QvuFj0nGgO970fySghXGmuw+Fd0gD2x3+MqCWLIPf5oxdv1Ka6b2q+z9RP01P/IaKPMEramy+7cNy/Lw8c3hw==" }, - "inquirer": { - "version": "7.3.3", - "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.3.3.tgz", - "integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==", + "marked-terminal": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/marked-terminal/-/marked-terminal-4.2.0.tgz", + "integrity": "sha512-DQfNRV9svZf0Dm9Cf5x5xaVJ1+XjxQW6XjFJ5HFkVyK52SDpj5PCBzS5X5r2w9nHr3mlB0T5201UMLue9fmhUw==", "dev": true, "requires": { - "ansi-escapes": "^4.2.1", + "ansi-escapes": "^4.3.1", + "cardinal": "^2.1.1", "chalk": "^4.1.0", - "cli-cursor": "^3.1.0", - "cli-width": "^3.0.0", - "external-editor": "^3.0.3", - "figures": "^3.0.0", - "lodash": "^4.17.19", - "mute-stream": "0.0.8", - "run-async": "^2.4.0", - "rxjs": "^6.6.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0", - "through": "^2.3.6" + "cli-table3": "^0.6.0", + "node-emoji": "^1.10.0", + "supports-hyperlinks": "^2.1.0" }, "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, "ansi-styles": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", @@ -5693,445 +14719,1311 @@ "requires": { "has-flag": "^4.0.0" } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + } } } }, - "internal-slot": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.3.tgz", - "integrity": "sha512-O0DB1JC/sPyZl7cIo78n5dR7eUSwwpYPiXRhTzNxZVAMUuB8vlnRFyLxdrVToks6XPLVnFfbzaVd5WLjhgg+vA==", - "dev": true, - "requires": { - "get-intrinsic": "^1.1.0", - "has": "^1.0.3", - "side-channel": "^1.0.4" - } - }, - "interpret": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/interpret/-/interpret-1.4.0.tgz", - "integrity": "sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==" - }, - "is-arrayish": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", - "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", - "dev": true - }, - "is-bigint": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", - "integrity": "sha512-zB9CruMamjym81i2JZ3UMn54PKGsQzsJeo6xvN3HJJ4CAsQNB6iRutp2To77OfCNuoxspsIhzaPoO1zyCEhFOg==", - "dev": true, - "requires": { - "has-bigints": "^1.0.1" - } + "mdurl": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/mdurl/-/mdurl-1.0.1.tgz", + "integrity": "sha1-/oWy7HWlkDfyrf7BAP1sYBdhFS4=" }, - "is-boolean-object": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/is-boolean-object/-/is-boolean-object-1.1.2.tgz", - "integrity": "sha512-gDYaKHJmnj4aWxyj6YHyXVpdQawtVLHU5cb+eztPGczf6cjuTdwve5ZIEfgXqH4e57An1D1AKf8CZ3kYrQRqYA==", + "mem-fs": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/mem-fs/-/mem-fs-1.2.0.tgz", + "integrity": "sha512-b8g0jWKdl8pM0LqAPdK9i8ERL7nYrzmJfRhxMiWH2uYdfYnb7uXnmwVb0ZGe7xyEl4lj+nLIU3yf4zPUT+XsVQ==", "dev": true, "requires": { - "call-bind": "^1.0.2", - "has-tostringtag": "^1.0.0" - } - }, - "is-callable": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.4.tgz", - "integrity": "sha512-nsuwtxZfMX67Oryl9LCQ+upnC0Z0BgpwntpS89m1H/TLF0zNfzfLMV/9Wa/6MZsj0acpEjAO0KF1xT6ZdLl95w==", - "dev": true - }, - "is-core-module": { - "version": "2.8.0", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.8.0.tgz", - "integrity": "sha512-vd15qHsaqrRL7dtH6QNuy0ndJmRDrS9HAM1CAiSifNUFv4x1a0CCVsj18hJ1mShxIG6T2i1sO78MkP56r0nYRw==", - "requires": { - "has": "^1.0.3" + "through2": "^3.0.0", + "vinyl": "^2.0.1", + "vinyl-file": "^3.0.0" } }, - "is-date-object": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.5.tgz", - "integrity": "sha512-9YQaSxsAiSwcvS33MBk3wTCVnWK+HhF8VZR2jRxehM16QcVOdHqPn4VPHmRK4lSr38n9JriurInLcP90xsYNfQ==", + "mem-fs-editor": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/mem-fs-editor/-/mem-fs-editor-6.0.0.tgz", + "integrity": "sha512-e0WfJAMm8Gv1mP5fEq/Blzy6Lt1VbLg7gNnZmZak7nhrBTibs+c6nQ4SKs/ZyJYHS1mFgDJeopsLAv7Ow0FMFg==", "dev": true, "requires": { - "has-tostringtag": "^1.0.0" + "commondir": "^1.0.1", + "deep-extend": "^0.6.0", + "ejs": "^2.6.1", + "glob": "^7.1.4", + "globby": "^9.2.0", + "isbinaryfile": "^4.0.0", + "mkdirp": "^0.5.0", + "multimatch": "^4.0.0", + "rimraf": "^2.6.3", + "through2": "^3.0.1", + "vinyl": "^2.2.0" + }, + "dependencies": { + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "dir-glob": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.2.2.tgz", + "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", + "dev": true, + "requires": { + "path-type": "^3.0.0" + } + }, + "ejs": { + "version": "2.7.4", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-2.7.4.tgz", + "integrity": "sha512-7vmuyh5+kuUyJKePhQfRQBhXV5Ce+RnaeeQArKu1EAMpL3WbgMt5WG6uQZpEVvYSSsxMXRKOewtDk9RaTKXRlA==", + "dev": true + }, + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "9.2.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-9.2.0.tgz", + "integrity": "sha512-ollPHROa5mcxDEkwg6bPt3QbEf4pDQSNtd6JPL1YvOvAo/7/0VAm9TccUeoTmarjPw4pfUthSCqcyfNB1I3ZSg==", + "dev": true, + "requires": { + "@types/glob": "^7.1.1", + "array-union": "^1.0.2", + "dir-glob": "^2.2.2", + "fast-glob": "^2.2.6", + "glob": "^7.1.3", + "ignore": "^4.0.3", + "pify": "^4.0.1", + "slash": "^2.0.0" + } + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + } + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + }, + "dependencies": { + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + } + } + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "slash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", + "dev": true + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + } + } } }, - "is-docker": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", - "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", - "dev": true - }, - "is-extglob": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", - "dev": true - }, - "is-fullwidth-code-point": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", "dev": true }, - "is-generator-fn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/is-generator-fn/-/is-generator-fn-2.1.0.tgz", - "integrity": "sha512-cTIB4yPYL/Grw0EaSzASzg6bBy9gqCofvWN8okThAYIxKJZC+udlRAmGbM0XLeniEJSs8uEgHPGuHSe1XsOLSQ==", + "merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", "dev": true }, - "is-glob": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", - "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "micromatch": { + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.5.tgz", + "integrity": "sha512-DMy+ERcEW2q8Z2Po+WNXuw3c5YaUSFjAO5GsJqfEl7UjvtIuFKO6ZrKvcItdy98dwFI2N1tg3zNIdKaQT+aNdA==", "dev": true, "requires": { - "is-extglob": "^2.1.1" + "braces": "^3.0.2", + "picomatch": "^2.3.1" } }, - "is-negative-zero": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", - "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "mime": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.6.0.tgz", + "integrity": "sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg==", "dev": true }, - "is-number": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", "dev": true }, - "is-number-object": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/is-number-object/-/is-number-object-1.0.6.tgz", - "integrity": "sha512-bEVOqiRcvo3zO1+G2lVMy+gkkEm9Yh7cDMRusKKu5ZJKPUYSJwICTKZrNKHA2EbSP0Tu0+6B/emsYNHZyn6K8g==", + "mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", "dev": true, "requires": { - "has-tostringtag": "^1.0.0" + "mime-db": "1.52.0" } }, - "is-potential-custom-element-name": { + "mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true + }, + "mimic-response": { "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-potential-custom-element-name/-/is-potential-custom-element-name-1.0.1.tgz", - "integrity": "sha512-bCYeRA2rVibKZd+s2625gGnGF/t7DSqDs4dP7CrLA1m7jKWz6pps0LpYLJN8Q64HtmPKJ1hrN3nzPNKFEKOUiQ==", + "resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-1.0.1.tgz", + "integrity": "sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ==", "dev": true }, - "is-regex": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.4.tgz", - "integrity": "sha512-kvRdxDsxZjhzUX07ZnLydzS1TU/TJlTUHHY4YLL87e37oUA49DfkLqgy+VjFocowy29cKvcSiu+kIv728jTTVg==", - "dev": true, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", "requires": { - "call-bind": "^1.0.2", - "has-tostringtag": "^1.0.0" + "brace-expansion": "^1.1.7" } }, - "is-shared-array-buffer": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.1.tgz", - "integrity": "sha512-IU0NmyknYZN0rChcKhRO1X8LYz5Isj/Fsqh8NJOSf+N/hCOTwy29F32Ik7a+QszE63IdvmwdTPDd6cZ5pg4cwA==", + "minimist": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.6.tgz", + "integrity": "sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q==", "dev": true }, - "is-stream": { + "minimisted": { "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-2.0.1.tgz", - "integrity": "sha512-hFoiJiTl63nn+kstHGBtewWSKnQLpyb155KHheA1l39uvtO9nWIop1p3udqPcUd/xbF1VLMO4n7OI6p7RbngDg==", - "dev": true + "resolved": "https://registry.npmjs.org/minimisted/-/minimisted-2.0.1.tgz", + "integrity": "sha512-1oPjfuLQa2caorJUM8HV8lGgWCc0qqAO1MNv/k05G4qslmsndV/5WdNZrqCiyqiz3wohia2Ij2B7w2Dr7/IyrA==", + "dev": true, + "requires": { + "minimist": "^1.2.5" + } }, - "is-string": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/is-string/-/is-string-1.0.7.tgz", - "integrity": "sha512-tE2UXzivje6ofPW7l23cjDOMa09gb7xlAqG6jG5ej6uPV32TlWP3NKPigtaGeHNu9fohccRYvIiZMfOOnOYUtg==", + "minipass": { + "version": "2.9.0", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz", + "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", "dev": true, "requires": { - "has-tostringtag": "^1.0.0" + "safe-buffer": "^5.1.2", + "yallist": "^3.0.0" } }, - "is-symbol": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.4.tgz", - "integrity": "sha512-C/CPBqKWnvdcxqIARxyOh4v1UUEOCHpgDa0WYgpKDFMszcrPcffg5uhwSgPCLD2WWxmq6isisz87tzT01tuGhg==", + "minizlib": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz", + "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", "dev": true, "requires": { - "has-symbols": "^1.0.2" + "minipass": "^2.9.0" } }, - "is-typedarray": { + "mixin-deep": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", + "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", + "dev": true, + "requires": { + "for-in": "^1.0.2", + "is-extendable": "^1.0.1" + }, + "dependencies": { + "is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, + "requires": { + "is-plain-object": "^2.0.4" + } + } + } + }, + "mkdirp": { + "version": "0.5.6", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.6.tgz", + "integrity": "sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==", + "dev": true, + "requires": { + "minimist": "^1.2.6" + } + }, + "mkdirp-classic": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz", + "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==", + "dev": true + }, + "mkdirp2": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/mkdirp2/-/mkdirp2-1.0.5.tgz", + "integrity": "sha512-xOE9xbICroUDmG1ye2h4bZ8WBie9EGmACaco8K8cx6RlkJJrxGIqjGqztAI+NMhexXBcdGbSEzI6N3EJPevxZw==", + "dev": true + }, + "mock-stdin": { "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", - "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", + "resolved": "https://registry.npmjs.org/mock-stdin/-/mock-stdin-1.0.0.tgz", + "integrity": "sha512-tukRdb9Beu27t6dN+XztSRHq9J0B/CoAOySGzHfn8UTfmqipA5yNT/sDUEyYdAV3Hpka6Wx6kOMxuObdOex60Q==", "dev": true }, - "is-weakref": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", - "integrity": "sha512-qctsuLZmIQ0+vSSMfoVvyFe2+GSEvnmZ2ezTup1SBse9+twCCeial6EEi3Nc2KFcf6+qz2FBPnjXsk8xhKSaPQ==", + "moment": { + "version": "2.29.2", + "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.2.tgz", + "integrity": "sha512-UgzG4rvxYpN15jgCmVJwac49h9ly9NurikMWGPdVxm8GZD6XjkKPxDTjQQ43gtGgnV3X0cAyWDdP2Wexoquifg==", + "dev": true + }, + "ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true + }, + "multimatch": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/multimatch/-/multimatch-4.0.0.tgz", + "integrity": "sha512-lDmx79y1z6i7RNx0ZGCPq1bzJ6ZoDDKbvh7jxr9SJcWLkShMzXrHbYVpTdnhNM5MXpDUxCQ4DgqVttVXlBgiBQ==", "dev": true, "requires": { - "call-bind": "^1.0.2" + "@types/minimatch": "^3.0.3", + "array-differ": "^3.0.0", + "array-union": "^2.1.0", + "arrify": "^2.0.1", + "minimatch": "^3.0.4" + }, + "dependencies": { + "arrify": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-2.0.1.tgz", + "integrity": "sha512-3duEwti880xqi4eAMN8AyR4a0ByT90zoYdLlevfrvU43vb0YZwZVfxOgxWrLXXXpyugL0hNZc9G6BiB5B3nUug==", + "dev": true + } } }, - "is-wsl": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", - "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "multistream": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/multistream/-/multistream-2.1.1.tgz", + "integrity": "sha512-xasv76hl6nr1dEy3lPvy7Ej7K/Lx3O/FCvwge8PeVJpciPPoNCbaANcNiBug3IpdvTveZUcAV0DJzdnUDMesNQ==", "dev": true, "requires": { - "is-docker": "^2.0.0" + "inherits": "^2.0.1", + "readable-stream": "^2.0.5" + }, + "dependencies": { + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } } }, - "is-zip": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-zip/-/is-zip-1.0.0.tgz", - "integrity": "sha1-R7Co/004p2QxzP2ZqOFaTIa6IyU=", + "mustache": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/mustache/-/mustache-4.2.0.tgz", + "integrity": "sha512-71ippSywq5Yb7/tVYyGbkBggbU8H3u5Rz56fH60jGFgr8uHwxs+aSKeqmluIVzM0m0kB7xQjKS6qPfd0b2ZoqQ==", "dev": true }, - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", "dev": true }, - "isbinaryfile": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/isbinaryfile/-/isbinaryfile-4.0.8.tgz", - "integrity": "sha512-53h6XFniq77YdW+spoRrebh0mnmTxRPTlcuIArO57lmMdq4uBKFKaeTjnb92oYWrSn/LVL+LT+Hap2tFQj8V+w==", + "mv": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/mv/-/mv-2.1.1.tgz", + "integrity": "sha1-rmzg1vbV4KT32JN5jQPB6pVZtqI=", + "dev": true, + "optional": true, + "requires": { + "mkdirp": "~0.5.1", + "ncp": "~2.0.0", + "rimraf": "~2.4.0" + } + }, + "nan": { + "version": "2.15.0", + "resolved": "https://registry.npmjs.org/nan/-/nan-2.15.0.tgz", + "integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ==", + "dev": true, + "optional": true + }, + "nanoid": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.2.0.tgz", + "integrity": "sha512-fmsZYa9lpn69Ad5eDn7FMcnnSR+8R34W9qJEijxYhTbfOWzr22n1QxCMzXLK+ODyW2973V3Fux959iQoUxzUIA==" + }, + "nanomatch": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", + "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "fragment-cache": "^0.2.1", + "is-windows": "^1.0.2", + "kind-of": "^6.0.2", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + } + }, + "natural-compare": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", + "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", "dev": true }, - "isexe": { + "natural-orderby": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/natural-orderby/-/natural-orderby-2.0.3.tgz", + "integrity": "sha512-p7KTHxU0CUrcOXe62Zfrb5Z13nLvPhSWR/so3kFulUQU0sgUll2Z0LwpsLN351eOOD+hRGu/F1g+6xDfPeD++Q==", + "dev": true + }, + "ncp": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", - "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", + "resolved": "https://registry.npmjs.org/ncp/-/ncp-2.0.0.tgz", + "integrity": "sha1-GVoh1sRuNh0vsSgbo4uR6d9727M=", + "dev": true, + "optional": true + }, + "neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", "dev": true }, - "isstream": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", - "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", + "netmask": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/netmask/-/netmask-2.0.2.tgz", + "integrity": "sha512-dBpDMdxv9Irdq66304OLfEmQ9tbNRFnFTuZiLo+bD+r332bBmMJ8GBLXklIXXgxd3+v9+KUnZaUR5PJMa75Gsg==", "dev": true }, - "istanbul-lib-coverage": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.2.0.tgz", - "integrity": "sha512-eOeJ5BHCmHYvQK7xt9GkdHuzuCGS1Y6g9Gvnx3Ym33fz/HpLRYxiS0wHNr+m/MBC8B647Xt608vCDEvhl9c6Mw==", + "netrc-parser": { + "version": "3.1.6", + "resolved": "https://registry.npmjs.org/netrc-parser/-/netrc-parser-3.1.6.tgz", + "integrity": "sha512-lY+fmkqSwntAAjfP63jB4z5p5WbuZwyMCD3pInT7dpHU/Gc6Vv90SAC6A0aNiqaRGHiuZFBtiwu+pu8W/Eyotw==", + "dev": true, + "requires": { + "debug": "^3.1.0", + "execa": "^0.10.0" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "execa": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-0.10.0.tgz", + "integrity": "sha512-7XOMnz8Ynx1gGo/3hyV9loYNPWM94jG3+3T3Y8tsfSstFmETmENCMU/A/zj8Lyaj1lkgEepKepvd6240tBRvlw==", + "dev": true, + "requires": { + "cross-spawn": "^6.0.0", + "get-stream": "^3.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" + } + }, + "get-stream": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-3.0.0.tgz", + "integrity": "sha1-jpQ9E1jcN1VQVOy+LtsFqhdO3hQ=", + "dev": true + }, + "is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", + "dev": true + }, + "npm-run-path": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", + "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", + "dev": true, + "requires": { + "path-key": "^2.0.0" + } + } + } + }, + "nice-try": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", + "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", "dev": true }, - "istanbul-lib-instrument": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.3.tgz", - "integrity": "sha512-BXgQl9kf4WTCPCCpmFGoJkz/+uhvm7h7PFKUYxh7qarQd3ER33vHG//qaE8eN25l07YqZPpHXU9I09l/RD5aGQ==", + "no-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/no-case/-/no-case-3.0.4.tgz", + "integrity": "sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==", + "dev": true, + "requires": { + "lower-case": "^2.0.2", + "tslib": "^2.0.3" + } + }, + "nock": { + "version": "13.2.4", + "resolved": "https://registry.npmjs.org/nock/-/nock-13.2.4.tgz", + "integrity": "sha512-8GPznwxcPNCH/h8B+XZcKjYPXnUV5clOKCjAqyjsiqA++MpNx9E9+t8YPp0MbThO+KauRo7aZJ1WuIZmOrT2Ug==", + "dev": true, + "requires": { + "debug": "^4.1.0", + "json-stringify-safe": "^5.0.1", + "lodash.set": "^4.3.2", + "propagate": "^2.0.0" + } + }, + "node-emoji": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/node-emoji/-/node-emoji-1.11.0.tgz", + "integrity": "sha512-wo2DpQkQp7Sjm2A0cq+sN7EHKO6Sl0ctXeBdFZrL9T9+UywORbufTcTZxom8YqpLQt/FqNMUkOpkZrJVYSKD3A==", "dev": true, "requires": { - "@babel/core": "^7.7.5", - "@istanbuljs/schema": "^0.1.2", - "istanbul-lib-coverage": "^3.0.0", - "semver": "^6.3.0" + "lodash": "^4.17.21" } }, - "istanbul-lib-report": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", - "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", + "node-fetch": { + "version": "2.6.7", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.7.tgz", + "integrity": "sha512-ZjMPFEfVx5j+y2yF35Kzx5sF7kDzxuDj6ziH4FFbOp87zKDZNx8yExJIb05OGF4Nlt9IHFIMBkRl41VdvcNdbQ==", "dev": true, "requires": { - "istanbul-lib-coverage": "^3.0.0", - "make-dir": "^3.0.0", - "supports-color": "^7.1.0" + "whatwg-url": "^5.0.0" }, "dependencies": { - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=", "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=", + "dev": true + }, + "whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha1-lmRU6HZUYuN2RNNib2dCzotwll0=", "dev": true, "requires": { - "has-flag": "^4.0.0" + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" } } } }, - "istanbul-lib-source-maps": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", - "integrity": "sha512-n3s8EwkdFIJCG3BPKBYvskgXGoy88ARzvegkitk60NxRdwltLOTaH7CUiMRXvwYorl0Q712iEjcWB+fK/MrWVw==", + "node-int64": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", + "integrity": "sha1-h6kGXNs1XTGC2PlM4RGIuCXGijs=", + "dev": true + }, + "node-releases": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.3.tgz", + "integrity": "sha512-maHFz6OLqYxz+VQyCAtA3PTX4UP/53pa05fyDNc9CwjvJ0yEh6+xBwKsgCxMNhS8taUKBFYxfuiaD9U/55iFaw==", + "dev": true + }, + "node-stream-zip": { + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/node-stream-zip/-/node-stream-zip-1.13.2.tgz", + "integrity": "sha512-159NUO3quDIRVSVOQiQ/0VG8NFrGvVyUa+MUUleiOys3NLtrGtNkodqYYTRHU/kiXu/ygHZcNy/tZ4NUCcDUmQ==", + "dev": true + }, + "normalize-package-data": { + "version": "2.5.0", + "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", + "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", "dev": true, "requires": { - "debug": "^4.1.1", - "istanbul-lib-coverage": "^3.0.0", - "source-map": "^0.6.1" + "hosted-git-info": "^2.1.4", + "resolve": "^1.10.0", + "semver": "2 || 3 || 4 || 5", + "validate-npm-package-license": "^3.0.1" }, "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", "dev": true } } }, - "istanbul-reports": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.3.tgz", - "integrity": "sha512-x9LtDVtfm/t1GFiLl3NffC7hz+I1ragvgX1P/Lg1NlIagifZDKUkuuaAxH/qpwj2IuEfD8G2Bs/UKp+sZ/pKkg==", - "dev": true, - "requires": { - "html-escaper": "^2.0.0", - "istanbul-lib-report": "^3.0.0" - } + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true }, - "jest": { - "version": "27.0.6", - "resolved": "https://registry.npmjs.org/jest/-/jest-27.0.6.tgz", - "integrity": "sha512-EjV8aETrsD0wHl7CKMibKwQNQc3gIRBXlTikBmmHUeVMKaPFxdcUIBfoDqTSXDoGJIivAYGqCWVlzCSaVjPQsA==", + "normalize-url": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-6.1.0.tgz", + "integrity": "sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A==", + "dev": true + }, + "npm": { + "version": "7.24.2", + "resolved": "https://registry.npmjs.org/npm/-/npm-7.24.2.tgz", + "integrity": "sha512-120p116CE8VMMZ+hk8IAb1inCPk4Dj3VZw29/n2g6UI77urJKVYb7FZUDW8hY+EBnfsjI/2yrobBgFyzo7YpVQ==", "dev": true, "requires": { - "@jest/core": "^27.0.6", - "import-local": "^3.0.2", - "jest-cli": "^27.0.6" + "@isaacs/string-locale-compare": "^1.1.0", + "@npmcli/arborist": "^2.9.0", + "@npmcli/ci-detect": "^1.2.0", + "@npmcli/config": "^2.3.0", + "@npmcli/map-workspaces": "^1.0.4", + "@npmcli/package-json": "^1.0.1", + "@npmcli/run-script": "^1.8.6", + "abbrev": "~1.1.1", + "ansicolors": "~0.3.2", + "ansistyles": "~0.1.3", + "archy": "~1.0.0", + "cacache": "^15.3.0", + "chalk": "^4.1.2", + "chownr": "^2.0.0", + "cli-columns": "^3.1.2", + "cli-table3": "^0.6.0", + "columnify": "~1.5.4", + "fastest-levenshtein": "^1.0.12", + "glob": "^7.2.0", + "graceful-fs": "^4.2.8", + "hosted-git-info": "^4.0.2", + "ini": "^2.0.0", + "init-package-json": "^2.0.5", + "is-cidr": "^4.0.2", + "json-parse-even-better-errors": "^2.3.1", + "libnpmaccess": "^4.0.2", + "libnpmdiff": "^2.0.4", + "libnpmexec": "^2.0.1", + "libnpmfund": "^1.1.0", + "libnpmhook": "^6.0.2", + "libnpmorg": "^2.0.2", + "libnpmpack": "^2.0.1", + "libnpmpublish": "^4.0.1", + "libnpmsearch": "^3.1.1", + "libnpmteam": "^2.0.3", + "libnpmversion": "^1.2.1", + "make-fetch-happen": "^9.1.0", + "minipass": "^3.1.3", + "minipass-pipeline": "^1.2.4", + "mkdirp": "^1.0.4", + "mkdirp-infer-owner": "^2.0.0", + "ms": "^2.1.2", + "node-gyp": "^7.1.2", + "nopt": "^5.0.0", + "npm-audit-report": "^2.1.5", + "npm-install-checks": "^4.0.0", + "npm-package-arg": "^8.1.5", + "npm-pick-manifest": "^6.1.1", + "npm-profile": "^5.0.3", + "npm-registry-fetch": "^11.0.0", + "npm-user-validate": "^1.0.1", + "npmlog": "^5.0.1", + "opener": "^1.5.2", + "pacote": "^11.3.5", + "parse-conflict-json": "^1.1.1", + "qrcode-terminal": "^0.12.0", + "read": "~1.0.7", + "read-package-json": "^4.1.1", + "read-package-json-fast": "^2.0.3", + "readdir-scoped-modules": "^1.1.0", + "rimraf": "^3.0.2", + "semver": "^7.3.5", + "ssri": "^8.0.1", + "tar": "^6.1.11", + "text-table": "~0.2.0", + "tiny-relative-date": "^1.3.0", + "treeverse": "^1.0.4", + "validate-npm-package-name": "~3.0.0", + "which": "^2.0.2", + "write-file-atomic": "^3.0.3" }, "dependencies": { + "@gar/promisify": { + "version": "1.1.2", + "bundled": true, + "dev": true + }, + "@isaacs/string-locale-compare": { + "version": "1.1.0", + "bundled": true, + "dev": true + }, + "@npmcli/arborist": { + "version": "2.9.0", + "bundled": true, + "dev": true, + "requires": { + "@isaacs/string-locale-compare": "^1.0.1", + "@npmcli/installed-package-contents": "^1.0.7", + "@npmcli/map-workspaces": "^1.0.2", + "@npmcli/metavuln-calculator": "^1.1.0", + "@npmcli/move-file": "^1.1.0", + "@npmcli/name-from-folder": "^1.0.1", + "@npmcli/node-gyp": "^1.0.1", + "@npmcli/package-json": "^1.0.1", + "@npmcli/run-script": "^1.8.2", + "bin-links": "^2.2.1", + "cacache": "^15.0.3", + "common-ancestor-path": "^1.0.1", + "json-parse-even-better-errors": "^2.3.1", + "json-stringify-nice": "^1.1.4", + "mkdirp": "^1.0.4", + "mkdirp-infer-owner": "^2.0.0", + "npm-install-checks": "^4.0.0", + "npm-package-arg": "^8.1.5", + "npm-pick-manifest": "^6.1.0", + "npm-registry-fetch": "^11.0.0", + "pacote": "^11.3.5", + "parse-conflict-json": "^1.1.1", + "proc-log": "^1.0.0", + "promise-all-reject-late": "^1.0.0", + "promise-call-limit": "^1.0.1", + "read-package-json-fast": "^2.0.2", + "readdir-scoped-modules": "^1.1.0", + "rimraf": "^3.0.2", + "semver": "^7.3.5", + "ssri": "^8.0.1", + "treeverse": "^1.0.4", + "walk-up-path": "^1.0.0" + } + }, + "@npmcli/ci-detect": { + "version": "1.3.0", + "bundled": true, + "dev": true + }, + "@npmcli/config": { + "version": "2.3.0", + "bundled": true, + "dev": true, + "requires": { + "ini": "^2.0.0", + "mkdirp-infer-owner": "^2.0.0", + "nopt": "^5.0.0", + "semver": "^7.3.4", + "walk-up-path": "^1.0.0" + } + }, + "@npmcli/disparity-colors": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "requires": { + "ansi-styles": "^4.3.0" + } + }, + "@npmcli/fs": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "@gar/promisify": "^1.0.1", + "semver": "^7.3.5" + } + }, + "@npmcli/git": { + "version": "2.1.0", + "bundled": true, + "dev": true, + "requires": { + "@npmcli/promise-spawn": "^1.3.2", + "lru-cache": "^6.0.0", + "mkdirp": "^1.0.4", + "npm-pick-manifest": "^6.1.1", + "promise-inflight": "^1.0.1", + "promise-retry": "^2.0.1", + "semver": "^7.3.5", + "which": "^2.0.2" + } + }, + "@npmcli/installed-package-contents": { + "version": "1.0.7", + "bundled": true, + "dev": true, + "requires": { + "npm-bundled": "^1.1.1", + "npm-normalize-package-bin": "^1.0.1" + } + }, + "@npmcli/map-workspaces": { + "version": "1.0.4", + "bundled": true, + "dev": true, + "requires": { + "@npmcli/name-from-folder": "^1.0.1", + "glob": "^7.1.6", + "minimatch": "^3.0.4", + "read-package-json-fast": "^2.0.1" + } + }, + "@npmcli/metavuln-calculator": { + "version": "1.1.1", + "bundled": true, + "dev": true, + "requires": { + "cacache": "^15.0.5", + "pacote": "^11.1.11", + "semver": "^7.3.2" + } + }, + "@npmcli/move-file": { + "version": "1.1.2", + "bundled": true, + "dev": true, + "requires": { + "mkdirp": "^1.0.4", + "rimraf": "^3.0.2" + } + }, + "@npmcli/name-from-folder": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "@npmcli/node-gyp": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "@npmcli/package-json": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "requires": { + "json-parse-even-better-errors": "^2.3.1" + } + }, + "@npmcli/promise-spawn": { + "version": "1.3.2", + "bundled": true, + "dev": true, + "requires": { + "infer-owner": "^1.0.4" + } + }, + "@npmcli/run-script": { + "version": "1.8.6", + "bundled": true, + "dev": true, + "requires": { + "@npmcli/node-gyp": "^1.0.2", + "@npmcli/promise-spawn": "^1.3.2", + "node-gyp": "^7.1.0", + "read-package-json-fast": "^2.0.1" + } + }, + "@tootallnate/once": { + "version": "1.1.2", + "bundled": true, + "dev": true + }, + "abbrev": { + "version": "1.1.1", + "bundled": true, + "dev": true + }, + "agent-base": { + "version": "6.0.2", + "bundled": true, + "dev": true, + "requires": { + "debug": "4" + } + }, + "agentkeepalive": { + "version": "4.1.4", + "bundled": true, + "dev": true, + "requires": { + "debug": "^4.1.0", + "depd": "^1.1.2", + "humanize-ms": "^1.2.1" + } + }, + "aggregate-error": { + "version": "3.1.0", + "bundled": true, + "dev": true, + "requires": { + "clean-stack": "^2.0.0", + "indent-string": "^4.0.0" + } + }, + "ajv": { + "version": "6.12.6", + "bundled": true, + "dev": true, + "requires": { + "fast-deep-equal": "^3.1.1", + "fast-json-stable-stringify": "^2.0.0", + "json-schema-traverse": "^0.4.1", + "uri-js": "^4.2.2" + } + }, + "ansi-regex": { + "version": "2.1.1", + "bundled": true, + "dev": true + }, "ansi-styles": { "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "bundled": true, + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "ansicolors": { + "version": "0.3.2", + "bundled": true, + "dev": true + }, + "ansistyles": { + "version": "0.1.3", + "bundled": true, + "dev": true + }, + "aproba": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "archy": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "are-we-there-yet": { + "version": "1.1.6", + "bundled": true, + "dev": true, + "requires": { + "delegates": "^1.0.0", + "readable-stream": "^3.6.0" + } + }, + "asap": { + "version": "2.0.6", + "bundled": true, + "dev": true + }, + "asn1": { + "version": "0.2.4", + "bundled": true, + "dev": true, + "requires": { + "safer-buffer": "~2.1.0" + } + }, + "assert-plus": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "asynckit": { + "version": "0.4.0", + "bundled": true, + "dev": true + }, + "aws-sign2": { + "version": "0.7.0", + "bundled": true, + "dev": true + }, + "aws4": { + "version": "1.11.0", + "bundled": true, + "dev": true + }, + "balanced-match": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "bcrypt-pbkdf": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "tweetnacl": "^0.14.3" + } + }, + "bin-links": { + "version": "2.2.1", + "bundled": true, + "dev": true, + "requires": { + "cmd-shim": "^4.0.1", + "mkdirp": "^1.0.3", + "npm-normalize-package-bin": "^1.0.0", + "read-cmd-shim": "^2.0.0", + "rimraf": "^3.0.0", + "write-file-atomic": "^3.0.3" + } + }, + "binary-extensions": { + "version": "2.2.0", + "bundled": true, + "dev": true + }, + "brace-expansion": { + "version": "1.1.11", + "bundled": true, + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "builtins": { + "version": "1.0.3", + "bundled": true, + "dev": true + }, + "cacache": { + "version": "15.3.0", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "@npmcli/fs": "^1.0.0", + "@npmcli/move-file": "^1.0.1", + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "glob": "^7.1.4", + "infer-owner": "^1.0.4", + "lru-cache": "^6.0.0", + "minipass": "^3.1.1", + "minipass-collect": "^1.0.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.2", + "mkdirp": "^1.0.3", + "p-map": "^4.0.0", + "promise-inflight": "^1.0.1", + "rimraf": "^3.0.2", + "ssri": "^8.0.1", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" } }, + "caseless": { + "version": "0.12.0", + "bundled": true, + "dev": true + }, "chalk": { "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "bundled": true, "dev": true, "requires": { "ansi-styles": "^4.1.0", "supports-color": "^7.1.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "chownr": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "cidr-regex": { + "version": "3.1.1", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "ip-regex": "^4.1.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "clean-stack": { + "version": "2.2.0", + "bundled": true, "dev": true }, - "jest-cli": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-cli/-/jest-cli-27.4.5.tgz", - "integrity": "sha512-hrky3DSgE0u7sQxaCL7bdebEPHx5QzYmrGuUjaPLmPE8jx5adtvGuOlRspvMoVLTTDOHRnZDoRLYJuA+VCI7Hg==", + "cli-columns": { + "version": "3.1.2", + "bundled": true, "dev": true, "requires": { - "@jest/core": "^27.4.5", - "@jest/test-result": "^27.4.2", - "@jest/types": "^27.4.2", - "chalk": "^4.0.0", - "exit": "^0.1.2", - "graceful-fs": "^4.2.4", - "import-local": "^3.0.2", - "jest-config": "^27.4.5", - "jest-util": "^27.4.2", - "jest-validate": "^27.4.2", - "prompts": "^2.0.1", - "yargs": "^16.2.0" + "string-width": "^2.0.0", + "strip-ansi": "^3.0.1" } }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } - } - } - }, - "jest-changed-files": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-changed-files/-/jest-changed-files-27.4.2.tgz", - "integrity": "sha512-/9x8MjekuzUQoPjDHbBiXbNEBauhrPU2ct7m8TfCg69ywt1y/N+yYwGh3gCpnqUS3klYWDU/lSNgv+JhoD2k1A==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "execa": "^5.0.0", - "throat": "^6.0.1" - } - }, - "jest-circus": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-circus/-/jest-circus-27.4.5.tgz", - "integrity": "sha512-eTNWa9wsvBwPykhMMShheafbwyakcdHZaEYh5iRrQ0PFJxkDP/e3U/FvzGuKWu2WpwUA3C3hPlfpuzvOdTVqnw==", - "dev": true, - "requires": { - "@jest/environment": "^27.4.4", - "@jest/test-result": "^27.4.2", - "@jest/types": "^27.4.2", - "@types/node": "*", - "chalk": "^4.0.0", - "co": "^4.6.0", - "dedent": "^0.7.0", - "expect": "^27.4.2", - "is-generator-fn": "^2.0.0", - "jest-each": "^27.4.2", - "jest-matcher-utils": "^27.4.2", - "jest-message-util": "^27.4.2", - "jest-runtime": "^27.4.5", - "jest-snapshot": "^27.4.5", - "jest-util": "^27.4.2", - "pretty-format": "^27.4.2", - "slash": "^3.0.0", - "stack-utils": "^2.0.3", - "throat": "^6.0.1" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "cli-table3": { + "version": "0.6.0", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "colors": "^1.1.2", + "object-assign": "^4.1.0", + "string-width": "^4.2.0" + }, + "dependencies": { + "ansi-regex": { + "version": "5.0.0", + "bundled": true, + "dev": true + }, + "is-fullwidth-code-point": { + "version": "3.0.0", + "bundled": true, + "dev": true + }, + "string-width": { + "version": "4.2.2", + "bundled": true, + "dev": true, + "requires": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + } + }, + "strip-ansi": { + "version": "6.0.0", + "bundled": true, + "dev": true, + "requires": { + "ansi-regex": "^5.0.0" + } + } } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "clone": { + "version": "1.0.4", + "bundled": true, + "dev": true + }, + "cmd-shim": { + "version": "4.1.0", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "mkdirp-infer-owner": "^2.0.0" } }, + "code-point-at": { + "version": "1.1.0", + "bundled": true, + "dev": true + }, "color-convert": { "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "bundled": true, "dev": true, "requires": { "color-name": "~1.1.4" @@ -6139,2221 +16031,1554 @@ }, "color-name": { "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "color-support": { + "version": "1.1.3", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "colors": { + "version": "1.4.0", + "bundled": true, + "dev": true, + "optional": true + }, + "columnify": { + "version": "1.5.4", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "strip-ansi": "^3.0.0", + "wcwidth": "^1.0.0" } - } - } - }, - "jest-config": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-config/-/jest-config-27.4.5.tgz", - "integrity": "sha512-t+STVJtPt+fpqQ8GBw850NtSQbnDOw/UzdPfzDaHQ48/AylQlW7LHj3dH+ndxhC1UxJ0Q3qkq7IH+nM1skwTwA==", - "dev": true, - "requires": { - "@babel/core": "^7.1.0", - "@jest/test-sequencer": "^27.4.5", - "@jest/types": "^27.4.2", - "babel-jest": "^27.4.5", - "chalk": "^4.0.0", - "ci-info": "^3.2.0", - "deepmerge": "^4.2.2", - "glob": "^7.1.1", - "graceful-fs": "^4.2.4", - "jest-circus": "^27.4.5", - "jest-environment-jsdom": "^27.4.4", - "jest-environment-node": "^27.4.4", - "jest-get-type": "^27.4.0", - "jest-jasmine2": "^27.4.5", - "jest-regex-util": "^27.4.0", - "jest-resolve": "^27.4.5", - "jest-runner": "^27.4.5", - "jest-util": "^27.4.2", - "jest-validate": "^27.4.2", - "micromatch": "^4.0.4", - "pretty-format": "^27.4.2", - "slash": "^3.0.0" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "combined-stream": { + "version": "1.0.8", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "delayed-stream": "~1.0.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "common-ancestor-path": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "concat-map": { + "version": "0.0.1", + "bundled": true, + "dev": true + }, + "console-control-strings": { + "version": "1.1.0", + "bundled": true, + "dev": true + }, + "core-util-is": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "dashdash": { + "version": "1.14.1", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "assert-plus": "^1.0.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "debug": { + "version": "4.3.2", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "ms": "2.1.2" + }, + "dependencies": { + "ms": { + "version": "2.1.2", + "bundled": true, + "dev": true + } } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "debuglog": { + "version": "1.0.1", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "defaults": { + "version": "1.0.3", + "bundled": true, + "dev": true, + "requires": { + "clone": "^1.0.2" + } + }, + "delayed-stream": { + "version": "1.0.0", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "delegates": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "depd": { + "version": "1.1.2", + "bundled": true, + "dev": true + }, + "dezalgo": { + "version": "1.0.3", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "asap": "^2.0.0", + "wrappy": "1" } - } - } - }, - "jest-diff": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-diff/-/jest-diff-27.4.2.tgz", - "integrity": "sha512-ujc9ToyUZDh9KcqvQDkk/gkbf6zSaeEg9AiBxtttXW59H/AcqEYp1ciXAtJp+jXWva5nAf/ePtSsgWwE5mqp4Q==", - "dev": true, - "requires": { - "chalk": "^4.0.0", - "diff-sequences": "^27.4.0", - "jest-get-type": "^27.4.0", - "pretty-format": "^27.4.2" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "diff": { + "version": "5.0.0", + "bundled": true, + "dev": true + }, + "ecc-jsbn": { + "version": "0.1.2", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "emoji-regex": { + "version": "8.0.0", + "bundled": true, + "dev": true + }, + "encoding": { + "version": "0.1.13", + "bundled": true, "dev": true, + "optional": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "iconv-lite": "^0.6.2" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "env-paths": { + "version": "2.2.1", + "bundled": true, + "dev": true + }, + "err-code": { + "version": "2.0.3", + "bundled": true, + "dev": true + }, + "extend": { + "version": "3.0.2", + "bundled": true, + "dev": true + }, + "extsprintf": { + "version": "1.3.0", + "bundled": true, + "dev": true + }, + "fast-deep-equal": { + "version": "3.1.3", + "bundled": true, + "dev": true + }, + "fast-json-stable-stringify": { + "version": "2.1.0", + "bundled": true, + "dev": true + }, + "fastest-levenshtein": { + "version": "1.0.12", + "bundled": true, + "dev": true + }, + "forever-agent": { + "version": "0.6.1", + "bundled": true, + "dev": true + }, + "fs-minipass": { + "version": "2.1.0", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "minipass": "^3.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "fs.realpath": { + "version": "1.0.0", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "function-bind": { + "version": "1.1.1", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "gauge": { + "version": "3.0.1", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "aproba": "^1.0.3 || ^2.0.0", + "color-support": "^1.1.2", + "console-control-strings": "^1.0.0", + "has-unicode": "^2.0.1", + "object-assign": "^4.1.1", + "signal-exit": "^3.0.0", + "string-width": "^1.0.1 || ^2.0.0", + "strip-ansi": "^3.0.1 || ^4.0.0", + "wide-align": "^1.1.2" } - } - } - }, - "jest-docblock": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/jest-docblock/-/jest-docblock-27.4.0.tgz", - "integrity": "sha512-7TBazUdCKGV7svZ+gh7C8esAnweJoG+SvcF6Cjqj4l17zA2q1cMwx2JObSioubk317H+cjcHgP+7fTs60paulg==", - "dev": true, - "requires": { - "detect-newline": "^3.0.0" - } - }, - "jest-each": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-each/-/jest-each-27.4.2.tgz", - "integrity": "sha512-53V2MNyW28CTruB3lXaHNk6PkiIFuzdOC9gR3C6j8YE/ACfrPnz+slB0s17AgU1TtxNzLuHyvNlLJ+8QYw9nBg==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "chalk": "^4.0.0", - "jest-get-type": "^27.4.0", - "jest-util": "^27.4.2", - "pretty-format": "^27.4.2" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "getpass": { + "version": "0.1.7", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "assert-plus": "^1.0.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "glob": { + "version": "7.2.0", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "graceful-fs": { + "version": "4.2.8", + "bundled": true, + "dev": true + }, + "har-schema": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "har-validator": { + "version": "5.1.5", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "ajv": "^6.12.3", + "har-schema": "^2.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true + "has": { + "version": "1.0.3", + "bundled": true, + "dev": true, + "requires": { + "function-bind": "^1.1.1" + } }, "has-flag": { "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "has-unicode": { + "version": "2.0.1", + "bundled": true, + "dev": true + }, + "hosted-git-info": { + "version": "4.0.2", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "lru-cache": "^6.0.0" } - } - } - }, - "jest-environment-jsdom": { - "version": "27.4.4", - "resolved": "https://registry.npmjs.org/jest-environment-jsdom/-/jest-environment-jsdom-27.4.4.tgz", - "integrity": "sha512-cYR3ndNfHBqQgFvS1RL7dNqSvD//K56j/q1s2ygNHcfTCAp12zfIromO1w3COmXrxS8hWAh7+CmZmGCIoqGcGA==", - "dev": true, - "requires": { - "@jest/environment": "^27.4.4", - "@jest/fake-timers": "^27.4.2", - "@jest/types": "^27.4.2", - "@types/node": "*", - "jest-mock": "^27.4.2", - "jest-util": "^27.4.2", - "jsdom": "^16.6.0" - } - }, - "jest-environment-node": { - "version": "27.4.4", - "resolved": "https://registry.npmjs.org/jest-environment-node/-/jest-environment-node-27.4.4.tgz", - "integrity": "sha512-D+v3lbJ2GjQTQR23TK0kY3vFVmSeea05giInI41HHOaJnAwOnmUHTZgUaZL+VxUB43pIzoa7PMwWtCVlIUoVoA==", - "dev": true, - "requires": { - "@jest/environment": "^27.4.4", - "@jest/fake-timers": "^27.4.2", - "@jest/types": "^27.4.2", - "@types/node": "*", - "jest-mock": "^27.4.2", - "jest-util": "^27.4.2" - } - }, - "jest-get-type": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/jest-get-type/-/jest-get-type-27.4.0.tgz", - "integrity": "sha512-tk9o+ld5TWq41DkK14L4wox4s2D9MtTpKaAVzXfr5CUKm5ZK2ExcaFE0qls2W71zE/6R2TxxrK9w2r6svAFDBQ==", - "dev": true - }, - "jest-haste-map": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-haste-map/-/jest-haste-map-27.4.5.tgz", - "integrity": "sha512-oJm1b5qhhPs78K24EDGifWS0dELYxnoBiDhatT/FThgB9yxqUm5F6li3Pv+Q+apMBmmPNzOBnZ7ZxWMB1Leq1Q==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "@types/graceful-fs": "^4.1.2", - "@types/node": "*", - "anymatch": "^3.0.3", - "fb-watchman": "^2.0.0", - "fsevents": "^2.3.2", - "graceful-fs": "^4.2.4", - "jest-regex-util": "^27.4.0", - "jest-serializer": "^27.4.0", - "jest-util": "^27.4.2", - "jest-worker": "^27.4.5", - "micromatch": "^4.0.4", - "walker": "^1.0.7" - } - }, - "jest-jasmine2": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-jasmine2/-/jest-jasmine2-27.4.5.tgz", - "integrity": "sha512-oUnvwhJDj2LhOiUB1kdnJjkx8C5PwgUZQb9urF77mELH9DGR4e2GqpWQKBOYXWs5+uTN9BGDqRz3Aeg5Wts7aw==", - "dev": true, - "requires": { - "@babel/traverse": "^7.1.0", - "@jest/environment": "^27.4.4", - "@jest/source-map": "^27.4.0", - "@jest/test-result": "^27.4.2", - "@jest/types": "^27.4.2", - "@types/node": "*", - "chalk": "^4.0.0", - "co": "^4.6.0", - "expect": "^27.4.2", - "is-generator-fn": "^2.0.0", - "jest-each": "^27.4.2", - "jest-matcher-utils": "^27.4.2", - "jest-message-util": "^27.4.2", - "jest-runtime": "^27.4.5", - "jest-snapshot": "^27.4.5", - "jest-util": "^27.4.2", - "pretty-format": "^27.4.2", - "throat": "^6.0.1" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "http-cache-semantics": { + "version": "4.1.0", + "bundled": true, + "dev": true + }, + "http-proxy-agent": { + "version": "4.0.1", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "http-signature": { + "version": "1.2.0", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "https-proxy-agent": { + "version": "5.0.0", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "agent-base": "6", + "debug": "4" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true - }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "humanize-ms": { + "version": "1.2.1", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "ms": "^2.0.0" } - } - } - }, - "jest-leak-detector": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-leak-detector/-/jest-leak-detector-27.4.2.tgz", - "integrity": "sha512-ml0KvFYZllzPBJWDei3mDzUhyp/M4ubKebX++fPaudpe8OsxUE+m+P6ciVLboQsrzOCWDjE20/eXew9QMx/VGw==", - "dev": true, - "requires": { - "jest-get-type": "^27.4.0", - "pretty-format": "^27.4.2" - } - }, - "jest-matcher-utils": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-matcher-utils/-/jest-matcher-utils-27.4.2.tgz", - "integrity": "sha512-jyP28er3RRtMv+fmYC/PKG8wvAmfGcSNproVTW2Y0P/OY7/hWUOmsPfxN1jOhM+0u2xU984u2yEagGivz9OBGQ==", - "dev": true, - "requires": { - "chalk": "^4.0.0", - "jest-diff": "^27.4.2", - "jest-get-type": "^27.4.0", - "pretty-format": "^27.4.2" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "iconv-lite": { + "version": "0.6.3", + "bundled": true, "dev": true, + "optional": true, "requires": { - "color-convert": "^2.0.1" + "safer-buffer": ">= 2.1.2 < 3.0.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "ignore-walk": { + "version": "3.0.4", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "minimatch": "^3.0.4" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "imurmurhash": { + "version": "0.1.4", + "bundled": true, + "dev": true + }, + "indent-string": { + "version": "4.0.0", + "bundled": true, + "dev": true + }, + "infer-owner": { + "version": "1.0.4", + "bundled": true, + "dev": true + }, + "inflight": { + "version": "1.0.6", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "once": "^1.3.0", + "wrappy": "1" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "inherits": { + "version": "2.0.4", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "ini": { + "version": "2.0.0", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "init-package-json": { + "version": "2.0.5", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "npm-package-arg": "^8.1.5", + "promzard": "^0.3.0", + "read": "~1.0.1", + "read-package-json": "^4.1.1", + "semver": "^7.3.5", + "validate-npm-package-license": "^3.0.4", + "validate-npm-package-name": "^3.0.0" } - } - } - }, - "jest-message-util": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-message-util/-/jest-message-util-27.4.2.tgz", - "integrity": "sha512-OMRqRNd9E0DkBLZpFtZkAGYOXl6ZpoMtQJWTAREJKDOFa0M6ptB7L67tp+cszMBkvSgKOhNtQp2Vbcz3ZZKo/w==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.12.13", - "@jest/types": "^27.4.2", - "@types/stack-utils": "^2.0.0", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.4", - "micromatch": "^4.0.4", - "pretty-format": "^27.4.2", - "slash": "^3.0.0", - "stack-utils": "^2.0.3" - }, - "dependencies": { - "ansi-styles": { + }, + "ip": { + "version": "1.1.5", + "bundled": true, + "dev": true + }, + "ip-regex": { "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "bundled": true, + "dev": true + }, + "is-cidr": { + "version": "4.0.2", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "cidr-regex": "^3.1.1" + } + }, + "is-core-module": { + "version": "2.7.0", + "bundled": true, + "dev": true, + "requires": { + "has": "^1.0.3" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", - "dev": true, - "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" - } + "is-fullwidth-code-point": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "is-lambda": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "is-typedarray": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "isexe": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "isstream": { + "version": "0.1.2", + "bundled": true, + "dev": true + }, + "jsbn": { + "version": "0.1.1", + "bundled": true, + "dev": true + }, + "json-parse-even-better-errors": { + "version": "2.3.1", + "bundled": true, + "dev": true }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, - "requires": { - "color-name": "~1.1.4" - } + "json-schema": { + "version": "0.2.3", + "bundled": true, + "dev": true }, - "color-name": { + "json-schema-traverse": { + "version": "0.4.1", + "bundled": true, + "dev": true + }, + "json-stringify-nice": { "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "json-stringify-safe": { + "version": "5.0.1", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "jsonparse": { + "version": "1.3.1", + "bundled": true, + "dev": true + }, + "jsprim": { + "version": "1.4.1", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "assert-plus": "1.0.0", + "extsprintf": "1.3.0", + "json-schema": "0.2.3", + "verror": "1.10.0" } - } - } - }, - "jest-mock": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-mock/-/jest-mock-27.4.2.tgz", - "integrity": "sha512-PDDPuyhoukk20JrQKeofK12hqtSka7mWH0QQuxSNgrdiPsrnYYLS6wbzu/HDlxZRzji5ylLRULeuI/vmZZDrYA==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "@types/node": "*" - } - }, - "jest-pnp-resolver": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/jest-pnp-resolver/-/jest-pnp-resolver-1.2.2.tgz", - "integrity": "sha512-olV41bKSMm8BdnuMsewT4jqlZ8+3TCARAXjZGT9jcoSnrfUnRCqnMoF9XEeoWjbzObpqF9dRhHQj0Xb9QdF6/w==", - "dev": true - }, - "jest-regex-util": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/jest-regex-util/-/jest-regex-util-27.4.0.tgz", - "integrity": "sha512-WeCpMpNnqJYMQoOjm1nTtsgbR4XHAk1u00qDoNBQoykM280+/TmgA5Qh5giC1ecy6a5d4hbSsHzpBtu5yvlbEg==", - "dev": true - }, - "jest-resolve": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-resolve/-/jest-resolve-27.4.5.tgz", - "integrity": "sha512-xU3z1BuOz/hUhVUL+918KqUgK+skqOuUsAi7A+iwoUldK6/+PW+utK8l8cxIWT9AW7IAhGNXjSAh1UYmjULZZw==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "chalk": "^4.0.0", - "graceful-fs": "^4.2.4", - "jest-haste-map": "^27.4.5", - "jest-pnp-resolver": "^1.2.2", - "jest-util": "^27.4.2", - "jest-validate": "^27.4.2", - "resolve": "^1.20.0", - "resolve.exports": "^1.1.0", - "slash": "^3.0.0" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "just-diff": { + "version": "3.1.1", + "bundled": true, + "dev": true + }, + "just-diff-apply": { + "version": "3.0.0", + "bundled": true, + "dev": true + }, + "libnpmaccess": { + "version": "4.0.3", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "aproba": "^2.0.0", + "minipass": "^3.1.1", + "npm-package-arg": "^8.1.2", + "npm-registry-fetch": "^11.0.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "libnpmdiff": { + "version": "2.0.4", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "@npmcli/disparity-colors": "^1.0.1", + "@npmcli/installed-package-contents": "^1.0.7", + "binary-extensions": "^2.2.0", + "diff": "^5.0.0", + "minimatch": "^3.0.4", + "npm-package-arg": "^8.1.4", + "pacote": "^11.3.4", + "tar": "^6.1.0" } }, - "color-convert": { + "libnpmexec": { "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "@npmcli/arborist": "^2.3.0", + "@npmcli/ci-detect": "^1.3.0", + "@npmcli/run-script": "^1.8.4", + "chalk": "^4.1.0", + "mkdirp-infer-owner": "^2.0.0", + "npm-package-arg": "^8.1.2", + "pacote": "^11.3.1", + "proc-log": "^1.0.0", + "read": "^1.0.7", + "read-package-json-fast": "^2.0.2", + "walk-up-path": "^1.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true - }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "libnpmfund": { + "version": "1.1.0", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "@npmcli/arborist": "^2.5.0" } - } - } - }, - "jest-resolve-dependencies": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-resolve-dependencies/-/jest-resolve-dependencies-27.4.5.tgz", - "integrity": "sha512-elEVvkvRK51y037NshtEkEnukMBWvlPzZHiL847OrIljJ8yIsujD2GXRPqDXC4rEVKbcdsy7W0FxoZb4WmEs7w==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "jest-regex-util": "^27.4.0", - "jest-snapshot": "^27.4.5" - } - }, - "jest-runner": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-runner/-/jest-runner-27.4.5.tgz", - "integrity": "sha512-/irauncTfmY1WkTaRQGRWcyQLzK1g98GYG/8QvIPviHgO1Fqz1JYeEIsSfF+9mc/UTA6S+IIHFgKyvUrtiBIZg==", - "dev": true, - "requires": { - "@jest/console": "^27.4.2", - "@jest/environment": "^27.4.4", - "@jest/test-result": "^27.4.2", - "@jest/transform": "^27.4.5", - "@jest/types": "^27.4.2", - "@types/node": "*", - "chalk": "^4.0.0", - "emittery": "^0.8.1", - "exit": "^0.1.2", - "graceful-fs": "^4.2.4", - "jest-docblock": "^27.4.0", - "jest-environment-jsdom": "^27.4.4", - "jest-environment-node": "^27.4.4", - "jest-haste-map": "^27.4.5", - "jest-leak-detector": "^27.4.2", - "jest-message-util": "^27.4.2", - "jest-resolve": "^27.4.5", - "jest-runtime": "^27.4.5", - "jest-util": "^27.4.2", - "jest-worker": "^27.4.5", - "source-map-support": "^0.5.6", - "throat": "^6.0.1" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "libnpmhook": { + "version": "6.0.3", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "aproba": "^2.0.0", + "npm-registry-fetch": "^11.0.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "libnpmorg": { + "version": "2.0.3", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "aproba": "^2.0.0", + "npm-registry-fetch": "^11.0.0" } }, - "color-convert": { + "libnpmpack": { "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "@npmcli/run-script": "^1.8.3", + "npm-package-arg": "^8.1.0", + "pacote": "^11.2.6" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true + "libnpmpublish": { + "version": "4.0.2", + "bundled": true, + "dev": true, + "requires": { + "normalize-package-data": "^3.0.2", + "npm-package-arg": "^8.1.2", + "npm-registry-fetch": "^11.0.0", + "semver": "^7.1.3", + "ssri": "^8.0.1" + } }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true + "libnpmsearch": { + "version": "3.1.2", + "bundled": true, + "dev": true, + "requires": { + "npm-registry-fetch": "^11.0.0" + } }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "libnpmteam": { + "version": "2.0.4", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "aproba": "^2.0.0", + "npm-registry-fetch": "^11.0.0" } - } - } - }, - "jest-runtime": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-runtime/-/jest-runtime-27.4.5.tgz", - "integrity": "sha512-CIYqwuJQXHQtPd/idgrx4zgJ6iCb6uBjQq1RSAGQrw2S8XifDmoM1Ot8NRd80ooAm+ZNdHVwsktIMGlA1F1FAQ==", - "dev": true, - "requires": { - "@jest/console": "^27.4.2", - "@jest/environment": "^27.4.4", - "@jest/globals": "^27.4.4", - "@jest/source-map": "^27.4.0", - "@jest/test-result": "^27.4.2", - "@jest/transform": "^27.4.5", - "@jest/types": "^27.4.2", - "@types/yargs": "^16.0.0", - "chalk": "^4.0.0", - "cjs-module-lexer": "^1.0.0", - "collect-v8-coverage": "^1.0.0", - "execa": "^5.0.0", - "exit": "^0.1.2", - "glob": "^7.1.3", - "graceful-fs": "^4.2.4", - "jest-haste-map": "^27.4.5", - "jest-message-util": "^27.4.2", - "jest-mock": "^27.4.2", - "jest-regex-util": "^27.4.0", - "jest-resolve": "^27.4.5", - "jest-snapshot": "^27.4.5", - "jest-util": "^27.4.2", - "jest-validate": "^27.4.2", - "slash": "^3.0.0", - "strip-bom": "^4.0.0", - "yargs": "^16.2.0" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "libnpmversion": { + "version": "1.2.1", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "@npmcli/git": "^2.0.7", + "@npmcli/run-script": "^1.8.4", + "json-parse-even-better-errors": "^2.3.1", + "semver": "^7.3.5", + "stringify-package": "^1.0.1" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "lru-cache": { + "version": "6.0.0", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "yallist": "^4.0.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "make-fetch-happen": { + "version": "9.1.0", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "agentkeepalive": "^4.1.3", + "cacache": "^15.2.0", + "http-cache-semantics": "^4.1.0", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-lambda": "^1.0.1", + "lru-cache": "^6.0.0", + "minipass": "^3.1.3", + "minipass-collect": "^1.0.2", + "minipass-fetch": "^1.3.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.4", + "negotiator": "^0.6.2", + "promise-retry": "^2.0.1", + "socks-proxy-agent": "^6.0.0", + "ssri": "^8.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "mime-db": { + "version": "1.49.0", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true + "mime-types": { + "version": "2.1.32", + "bundled": true, + "dev": true, + "requires": { + "mime-db": "1.49.0" + } }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "minimatch": { + "version": "3.0.4", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "brace-expansion": "^1.1.7" } - } - } - }, - "jest-serializer": { - "version": "27.4.0", - "resolved": "https://registry.npmjs.org/jest-serializer/-/jest-serializer-27.4.0.tgz", - "integrity": "sha512-RDhpcn5f1JYTX2pvJAGDcnsNTnsV9bjYPU8xcV+xPwOXnUPOQwf4ZEuiU6G9H1UztH+OapMgu/ckEVwO87PwnQ==", - "dev": true, - "requires": { - "@types/node": "*", - "graceful-fs": "^4.2.4" - } - }, - "jest-snapshot": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-snapshot/-/jest-snapshot-27.4.5.tgz", - "integrity": "sha512-eCi/iM1YJFrJWiT9de4+RpWWWBqsHiYxFG9V9o/n0WXs6GpW4lUt4FAHAgFPTLPqCUVzrMQmSmTZSgQzwqR7IQ==", - "dev": true, - "requires": { - "@babel/core": "^7.7.2", - "@babel/generator": "^7.7.2", - "@babel/parser": "^7.7.2", - "@babel/plugin-syntax-typescript": "^7.7.2", - "@babel/traverse": "^7.7.2", - "@babel/types": "^7.0.0", - "@jest/transform": "^27.4.5", - "@jest/types": "^27.4.2", - "@types/babel__traverse": "^7.0.4", - "@types/prettier": "^2.1.5", - "babel-preset-current-node-syntax": "^1.0.0", - "chalk": "^4.0.0", - "expect": "^27.4.2", - "graceful-fs": "^4.2.4", - "jest-diff": "^27.4.2", - "jest-get-type": "^27.4.0", - "jest-haste-map": "^27.4.5", - "jest-matcher-utils": "^27.4.2", - "jest-message-util": "^27.4.2", - "jest-resolve": "^27.4.5", - "jest-util": "^27.4.2", - "natural-compare": "^1.4.0", - "pretty-format": "^27.4.2", - "semver": "^7.3.2" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "minipass": { + "version": "3.1.5", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "yallist": "^4.0.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "minipass-collect": { + "version": "1.0.2", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "minipass": "^3.0.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "minipass-fetch": { + "version": "1.4.1", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "encoding": "^0.1.12", + "minipass": "^3.1.0", + "minipass-sized": "^1.0.3", + "minizlib": "^2.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true + "minipass-flush": { + "version": "1.0.5", + "bundled": true, + "dev": true, + "requires": { + "minipass": "^3.0.0" + } }, - "semver": { - "version": "7.3.5", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", - "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "minipass-json-stream": { + "version": "1.0.1", + "bundled": true, "dev": true, "requires": { - "lru-cache": "^6.0.0" + "jsonparse": "^1.3.1", + "minipass": "^3.0.0" } }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "minipass-pipeline": { + "version": "1.2.4", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "minipass": "^3.0.0" } - } - } - }, - "jest-util": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-27.4.2.tgz", - "integrity": "sha512-YuxxpXU6nlMan9qyLuxHaMMOzXAl5aGZWCSzben5DhLHemYQxCc4YK+4L3ZrCutT8GPQ+ui9k5D8rUJoDioMnA==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "@types/node": "*", - "chalk": "^4.0.0", - "ci-info": "^3.2.0", - "graceful-fs": "^4.2.4", - "picomatch": "^2.2.3" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "minipass-sized": { + "version": "1.0.3", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "minipass": "^3.0.0" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "minizlib": { + "version": "2.1.2", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "minipass": "^3.0.0", + "yallist": "^4.0.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "mkdirp": { + "version": "1.0.4", + "bundled": true, + "dev": true + }, + "mkdirp-infer-owner": { + "version": "2.0.0", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "chownr": "^2.0.0", + "infer-owner": "^1.0.4", + "mkdirp": "^1.0.3" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "ms": { + "version": "2.1.3", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "mute-stream": { + "version": "0.0.8", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "negotiator": { + "version": "0.6.2", + "bundled": true, + "dev": true + }, + "node-gyp": { + "version": "7.1.2", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "env-paths": "^2.2.0", + "glob": "^7.1.4", + "graceful-fs": "^4.2.3", + "nopt": "^5.0.0", + "npmlog": "^4.1.2", + "request": "^2.88.2", + "rimraf": "^3.0.2", + "semver": "^7.3.2", + "tar": "^6.0.2", + "which": "^2.0.2" + }, + "dependencies": { + "aproba": { + "version": "1.2.0", + "bundled": true, + "dev": true + }, + "gauge": { + "version": "2.7.4", + "bundled": true, + "dev": true, + "requires": { + "aproba": "^1.0.3", + "console-control-strings": "^1.0.0", + "has-unicode": "^2.0.0", + "object-assign": "^4.1.0", + "signal-exit": "^3.0.0", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wide-align": "^1.1.0" + } + }, + "is-fullwidth-code-point": { + "version": "1.0.0", + "bundled": true, + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "npmlog": { + "version": "4.1.2", + "bundled": true, + "dev": true, + "requires": { + "are-we-there-yet": "~1.1.2", + "console-control-strings": "~1.1.0", + "gauge": "~2.7.3", + "set-blocking": "~2.0.0" + } + }, + "string-width": { + "version": "1.0.2", + "bundled": true, + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + } } - } - } - }, - "jest-validate": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-validate/-/jest-validate-27.4.2.tgz", - "integrity": "sha512-hWYsSUej+Fs8ZhOm5vhWzwSLmVaPAxRy+Mr+z5MzeaHm9AxUpXdoVMEW4R86y5gOobVfBsMFLk4Rb+QkiEpx1A==", - "dev": true, - "requires": { - "@jest/types": "^27.4.2", - "camelcase": "^6.2.0", - "chalk": "^4.0.0", - "jest-get-type": "^27.4.0", - "leven": "^3.1.0", - "pretty-format": "^27.4.2" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "nopt": { + "version": "5.0.0", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "abbrev": "1" } }, - "camelcase": { - "version": "6.3.0", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.3.0.tgz", - "integrity": "sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==", - "dev": true - }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "normalize-package-data": { + "version": "3.0.3", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "hosted-git-info": "^4.0.1", + "is-core-module": "^2.5.0", + "semver": "^7.3.4", + "validate-npm-package-license": "^3.0.1" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "npm-audit-report": { + "version": "2.1.5", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "chalk": "^4.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true + "npm-bundled": { + "version": "1.1.2", + "bundled": true, + "dev": true, + "requires": { + "npm-normalize-package-bin": "^1.0.1" + } }, - "has-flag": { + "npm-install-checks": { "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "bundled": true, + "dev": true, + "requires": { + "semver": "^7.1.1" + } + }, + "npm-normalize-package-bin": { + "version": "1.0.1", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "npm-package-arg": { + "version": "8.1.5", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "hosted-git-info": "^4.0.1", + "semver": "^7.3.4", + "validate-npm-package-name": "^3.0.0" } - } - } - }, - "jest-watcher": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/jest-watcher/-/jest-watcher-27.4.2.tgz", - "integrity": "sha512-NJvMVyyBeXfDezhWzUOCOYZrUmkSCiatpjpm+nFUid74OZEHk6aMLrZAukIiFDwdbqp6mTM6Ui1w4oc+8EobQg==", - "dev": true, - "requires": { - "@jest/test-result": "^27.4.2", - "@jest/types": "^27.4.2", - "@types/node": "*", - "ansi-escapes": "^4.2.1", - "chalk": "^4.0.0", - "jest-util": "^27.4.2", - "string-length": "^4.0.1" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "npm-packlist": { + "version": "2.2.2", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "glob": "^7.1.6", + "ignore-walk": "^3.0.3", + "npm-bundled": "^1.1.1", + "npm-normalize-package-bin": "^1.0.1" } }, - "chalk": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", - "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "npm-pick-manifest": { + "version": "6.1.1", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.1.0", - "supports-color": "^7.1.0" + "npm-install-checks": "^4.0.0", + "npm-normalize-package-bin": "^1.0.1", + "npm-package-arg": "^8.1.2", + "semver": "^7.3.4" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "npm-profile": { + "version": "5.0.4", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "npm-registry-fetch": "^11.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "npm-registry-fetch": { + "version": "11.0.0", + "bundled": true, + "dev": true, + "requires": { + "make-fetch-happen": "^9.0.1", + "minipass": "^3.1.3", + "minipass-fetch": "^1.3.0", + "minipass-json-stream": "^1.0.1", + "minizlib": "^2.0.0", + "npm-package-arg": "^8.0.0" + } + }, + "npm-user-validate": { + "version": "1.0.1", + "bundled": true, "dev": true }, - "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "npmlog": { + "version": "5.0.1", + "bundled": true, + "dev": true, + "requires": { + "are-we-there-yet": "^2.0.0", + "console-control-strings": "^1.1.0", + "gauge": "^3.0.0", + "set-blocking": "^2.0.0" + }, + "dependencies": { + "are-we-there-yet": { + "version": "2.0.0", + "bundled": true, + "dev": true, + "requires": { + "delegates": "^1.0.0", + "readable-stream": "^3.6.0" + } + } + } + }, + "number-is-nan": { + "version": "1.0.1", + "bundled": true, "dev": true }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "oauth-sign": { + "version": "0.9.0", + "bundled": true, + "dev": true + }, + "object-assign": { + "version": "4.1.1", + "bundled": true, + "dev": true + }, + "once": { + "version": "1.4.0", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "wrappy": "1" } - } - } - }, - "jest-worker": { - "version": "27.4.5", - "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-27.4.5.tgz", - "integrity": "sha512-f2s8kEdy15cv9r7q4KkzGXvlY0JTcmCbMHZBfSQDwW77REr45IDWwd0lksDFeVHH2jJ5pqb90T77XscrjeGzzg==", - "dev": true, - "requires": { - "@types/node": "*", - "merge-stream": "^2.0.0", - "supports-color": "^8.0.0" - }, - "dependencies": { - "has-flag": { + }, + "opener": { + "version": "1.5.2", + "bundled": true, + "dev": true + }, + "p-map": { "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "bundled": true, + "dev": true, + "requires": { + "aggregate-error": "^3.0.0" + } + }, + "pacote": { + "version": "11.3.5", + "bundled": true, + "dev": true, + "requires": { + "@npmcli/git": "^2.1.0", + "@npmcli/installed-package-contents": "^1.0.6", + "@npmcli/promise-spawn": "^1.2.0", + "@npmcli/run-script": "^1.8.2", + "cacache": "^15.0.5", + "chownr": "^2.0.0", + "fs-minipass": "^2.1.0", + "infer-owner": "^1.0.4", + "minipass": "^3.1.3", + "mkdirp": "^1.0.3", + "npm-package-arg": "^8.0.1", + "npm-packlist": "^2.1.4", + "npm-pick-manifest": "^6.0.0", + "npm-registry-fetch": "^11.0.0", + "promise-retry": "^2.0.1", + "read-package-json-fast": "^2.0.1", + "rimraf": "^3.0.2", + "ssri": "^8.0.1", + "tar": "^6.1.0" + } + }, + "parse-conflict-json": { + "version": "1.1.1", + "bundled": true, + "dev": true, + "requires": { + "json-parse-even-better-errors": "^2.3.0", + "just-diff": "^3.0.1", + "just-diff-apply": "^3.0.0" + } + }, + "path-is-absolute": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "performance-now": { + "version": "2.1.0", + "bundled": true, "dev": true }, - "supports-color": { - "version": "8.1.1", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", - "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "proc-log": { + "version": "1.0.0", + "bundled": true, + "dev": true + }, + "promise-all-reject-late": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "promise-call-limit": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "promise-inflight": { + "version": "1.0.1", + "bundled": true, + "dev": true + }, + "promise-retry": { + "version": "2.0.1", + "bundled": true, "dev": true, "requires": { - "has-flag": "^4.0.0" + "err-code": "^2.0.2", + "retry": "^0.12.0" } - } - } - }, - "joi": { - "version": "17.5.0", - "resolved": "https://registry.npmjs.org/joi/-/joi-17.5.0.tgz", - "integrity": "sha512-R7hR50COp7StzLnDi4ywOXHrBrgNXuUUfJWIR5lPY5Bm/pOD3jZaTwpluUXVLRWcoWZxkrHBBJ5hLxgnlehbdw==", - "dev": true, - "requires": { - "@hapi/hoek": "^9.0.0", - "@hapi/topo": "^5.0.0", - "@sideway/address": "^4.1.3", - "@sideway/formula": "^3.0.0", - "@sideway/pinpoint": "^2.0.0" - } - }, - "js-tokens": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", - "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", - "dev": true - }, - "js-yaml": { - "version": "3.14.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", - "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", - "dev": true, - "requires": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" - } - }, - "js2xmlparser": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/js2xmlparser/-/js2xmlparser-4.0.2.tgz", - "integrity": "sha512-6n4D8gLlLf1n5mNLQPRfViYzu9RATblzPEtm1SthMX1Pjao0r9YI9nw7ZIfRxQMERS87mcswrg+r/OYrPRX6jA==", - "dev": true, - "requires": { - "xmlcreate": "^2.0.4" - } - }, - "jsbn": { - "version": "0.1.1", - "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", - "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", - "dev": true - }, - "jsdoc": { - "version": "3.6.7", - "resolved": "https://registry.npmjs.org/jsdoc/-/jsdoc-3.6.7.tgz", - "integrity": "sha512-sxKt7h0vzCd+3Y81Ey2qinupL6DpRSZJclS04ugHDNmRUXGzqicMJ6iwayhSA0S0DwwX30c5ozyUthr1QKF6uw==", - "dev": true, - "requires": { - "@babel/parser": "^7.9.4", - "bluebird": "^3.7.2", - "catharsis": "^0.9.0", - "escape-string-regexp": "^2.0.0", - "js2xmlparser": "^4.0.1", - "klaw": "^3.0.0", - "markdown-it": "^10.0.0", - "markdown-it-anchor": "^5.2.7", - "marked": "^2.0.3", - "mkdirp": "^1.0.4", - "requizzle": "^0.2.3", - "strip-json-comments": "^3.1.0", - "taffydb": "2.6.2", - "underscore": "~1.13.1" - }, - "dependencies": { - "escape-string-regexp": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", - "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", - "dev": true }, - "markdown-it": { - "version": "10.0.0", - "resolved": "https://registry.npmjs.org/markdown-it/-/markdown-it-10.0.0.tgz", - "integrity": "sha512-YWOP1j7UbDNz+TumYP1kpwnP0aEa711cJjrAQrzd0UXlbJfc5aAq0F/PZHjiioqDC1NKgvIMX+o+9Bk7yuM2dg==", + "promzard": { + "version": "0.3.0", + "bundled": true, "dev": true, "requires": { - "argparse": "^1.0.7", - "entities": "~2.0.0", - "linkify-it": "^2.0.0", - "mdurl": "^1.0.1", - "uc.micro": "^1.0.5" + "read": "1" } }, - "marked": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/marked/-/marked-2.1.3.tgz", - "integrity": "sha512-/Q+7MGzaETqifOMWYEA7HVMaZb4XbcRfaOzcSsHZEith83KGlvaSG33u0SKu89Mj5h+T8V2hM+8O45Qc5XTgwA==", + "psl": { + "version": "1.8.0", + "bundled": true, "dev": true - } - } - }, - "jsdoc-api": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/jsdoc-api/-/jsdoc-api-7.1.0.tgz", - "integrity": "sha512-yjIiZa6LFOgd0dyFW/R+3unnVUhhbU1CeBhisgjBPRHkar83rkgDtTMRdgQotSvt+pGlmknZqfwR5AQuMh9/6w==", - "dev": true, - "requires": { - "array-back": "^6.2.0", - "cache-point": "^2.0.0", - "collect-all": "^1.0.4", - "file-set": "^4.0.2", - "fs-then-native": "^2.0.0", - "jsdoc": "^3.6.7", - "object-to-spawn-args": "^2.0.1", - "temp-path": "^1.0.0", - "walk-back": "^5.1.0" - } - }, - "jsdoc-parse": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/jsdoc-parse/-/jsdoc-parse-6.0.1.tgz", - "integrity": "sha512-ij3Az5y2dp+ajMxYnEJH7kjKK5v6+yZ3Cg/KtRdoT15pIm6qTk/W8q72QdNLZ9jQm/U2/ifENFXXTOe6xIxGeA==", - "dev": true, - "requires": { - "array-back": "^6.1.1", - "lodash.omit": "^4.5.0", - "lodash.pick": "^4.4.0", - "reduce-extract": "^1.0.0", - "sort-array": "^4.1.4", - "test-value": "^3.0.0" - } - }, - "jsdoc-to-markdown": { - "version": "7.1.0", - "resolved": "https://registry.npmjs.org/jsdoc-to-markdown/-/jsdoc-to-markdown-7.1.0.tgz", - "integrity": "sha512-LJAiwrUaOpPqOmllqnVVqfBZh6KI/rHHfSwL7DerTpjLQWHfpndz/JUNlF5ngYjbL4aHNf7uJ1TuXl6xGfq5rg==", - "dev": true, - "requires": { - "array-back": "^6.2.0", - "command-line-tool": "^0.8.0", - "config-master": "^3.1.0", - "dmd": "^6.0.0", - "jsdoc-api": "^7.1.0", - "jsdoc-parse": "^6.0.1", - "walk-back": "^5.1.0" - } - }, - "jsdom": { - "version": "16.7.0", - "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-16.7.0.tgz", - "integrity": "sha512-u9Smc2G1USStM+s/x1ru5Sxrl6mPYCbByG1U/hUmqaVsm4tbNyS7CicOSRyuGQYZhTu0h84qkZZQ/I+dzizSVw==", - "dev": true, - "requires": { - "abab": "^2.0.5", - "acorn": "^8.2.4", - "acorn-globals": "^6.0.0", - "cssom": "^0.4.4", - "cssstyle": "^2.3.0", - "data-urls": "^2.0.0", - "decimal.js": "^10.2.1", - "domexception": "^2.0.1", - "escodegen": "^2.0.0", - "form-data": "^3.0.0", - "html-encoding-sniffer": "^2.0.1", - "http-proxy-agent": "^4.0.1", - "https-proxy-agent": "^5.0.0", - "is-potential-custom-element-name": "^1.0.1", - "nwsapi": "^2.2.0", - "parse5": "6.0.1", - "saxes": "^5.0.1", - "symbol-tree": "^3.2.4", - "tough-cookie": "^4.0.0", - "w3c-hr-time": "^1.0.2", - "w3c-xmlserializer": "^2.0.0", - "webidl-conversions": "^6.1.0", - "whatwg-encoding": "^1.0.5", - "whatwg-mimetype": "^2.3.0", - "whatwg-url": "^8.5.0", - "ws": "^7.4.6", - "xml-name-validator": "^3.0.0" - } - }, - "jsen": { - "version": "0.6.6", - "resolved": "https://registry.npmjs.org/jsen/-/jsen-0.6.6.tgz", - "integrity": "sha1-AkDBjPETUKwCFFb0in6xO9Z+BCA=", - "dev": true - }, - "jsesc": { - "version": "2.5.2", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", - "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", - "dev": true - }, - "jsforce": { - "version": "1.11.0", - "resolved": "https://registry.npmjs.org/jsforce/-/jsforce-1.11.0.tgz", - "integrity": "sha512-vYNXJXXdz9ZQNdfRqq/MCJ/zU7JGA7iEduwafQDzChR9FeqXgTNfHTppLVbw9mIniKkQZemmxSOtl7N04lj/5Q==", - "dev": true, - "requires": { - "base64-url": "^2.2.0", - "co-prompt": "^1.0.0", - "coffeescript": "^1.10.0", - "commander": "^2.9.0", - "csv-parse": "^4.10.1", - "csv-stringify": "^1.0.4", - "faye": "^1.4.0", - "inherits": "^2.0.1", - "lodash": "^4.17.19", - "multistream": "^2.0.5", - "opn": "^5.3.0", - "promise": "^7.1.1", - "readable-stream": "^2.1.0", - "request": "^2.72.0", - "xml2js": "^0.4.16" - } - }, - "json-parse-even-better-errors": { - "version": "2.3.1", - "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", - "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", - "dev": true - }, - "json-schema": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.4.0.tgz", - "integrity": "sha512-es94M3nTIfsEPisRafak+HDLfHXnKBhV3vU5eqPcS3flIWqcxJWgXHXiey3YrpaNsanY5ei1VoYEbOzijuq9BA==", - "dev": true - }, - "json-schema-traverse": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", - "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", - "dev": true - }, - "json-stable-stringify-without-jsonify": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz", - "integrity": "sha1-nbe1lJatPzz+8wp1FC0tkwrXJlE=", - "dev": true - }, - "json-stringify-safe": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", - "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", - "dev": true - }, - "json5": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/json5/-/json5-2.2.0.tgz", - "integrity": "sha512-f+8cldu7X/y7RAJurMEJmdoKXGB/X550w2Nr3tTbezL6RwEE/iMcm+tZnXeoZtKuOq6ft8+CqzEkrIgx1fPoQA==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, - "jsonfile": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", - "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.6" - } - }, - "jsonwebtoken": { - "version": "8.5.0", - "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-8.5.0.tgz", - "integrity": "sha512-IqEycp0znWHNA11TpYi77bVgyBO/pGESDh7Ajhas+u0ttkGkKYIIAjniL4Bw5+oVejVF+SYkaI7XKfwCCyeTuA==", - "dev": true, - "requires": { - "jws": "^3.2.1", - "lodash.includes": "^4.3.0", - "lodash.isboolean": "^3.0.3", - "lodash.isinteger": "^4.0.4", - "lodash.isnumber": "^3.0.3", - "lodash.isplainobject": "^4.0.6", - "lodash.isstring": "^4.0.1", - "lodash.once": "^4.0.0", - "ms": "^2.1.1", - "semver": "^5.6.0" - }, - "dependencies": { + }, + "punycode": { + "version": "2.1.1", + "bundled": true, + "dev": true + }, + "qrcode-terminal": { + "version": "0.12.0", + "bundled": true, + "dev": true + }, + "qs": { + "version": "6.5.2", + "bundled": true, + "dev": true + }, + "read": { + "version": "1.0.7", + "bundled": true, + "dev": true, + "requires": { + "mute-stream": "~0.0.4" + } + }, + "read-cmd-shim": { + "version": "2.0.0", + "bundled": true, + "dev": true + }, + "read-package-json": { + "version": "4.1.1", + "bundled": true, + "dev": true, + "requires": { + "glob": "^7.1.1", + "json-parse-even-better-errors": "^2.3.0", + "normalize-package-data": "^3.0.0", + "npm-normalize-package-bin": "^1.0.0" + } + }, + "read-package-json-fast": { + "version": "2.0.3", + "bundled": true, + "dev": true, + "requires": { + "json-parse-even-better-errors": "^2.3.0", + "npm-normalize-package-bin": "^1.0.1" + } + }, + "readable-stream": { + "version": "3.6.0", + "bundled": true, + "dev": true, + "requires": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + } + }, + "readdir-scoped-modules": { + "version": "1.1.0", + "bundled": true, + "dev": true, + "requires": { + "debuglog": "^1.0.1", + "dezalgo": "^1.0.0", + "graceful-fs": "^4.1.2", + "once": "^1.3.0" + } + }, + "request": { + "version": "2.88.2", + "bundled": true, + "dev": true, + "requires": { + "aws-sign2": "~0.7.0", + "aws4": "^1.8.0", + "caseless": "~0.12.0", + "combined-stream": "~1.0.6", + "extend": "~3.0.2", + "forever-agent": "~0.6.1", + "form-data": "~2.3.2", + "har-validator": "~5.1.3", + "http-signature": "~1.2.0", + "is-typedarray": "~1.0.0", + "isstream": "~0.1.2", + "json-stringify-safe": "~5.0.1", + "mime-types": "~2.1.19", + "oauth-sign": "~0.9.0", + "performance-now": "^2.1.0", + "qs": "~6.5.2", + "safe-buffer": "^5.1.2", + "tough-cookie": "~2.5.0", + "tunnel-agent": "^0.6.0", + "uuid": "^3.3.2" + }, + "dependencies": { + "form-data": { + "version": "2.3.3", + "bundled": true, + "dev": true, + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + } + }, + "tough-cookie": { + "version": "2.5.0", + "bundled": true, + "dev": true, + "requires": { + "psl": "^1.1.28", + "punycode": "^2.1.1" + } + } + } + }, + "retry": { + "version": "0.12.0", + "bundled": true, + "dev": true + }, + "rimraf": { + "version": "3.0.2", + "bundled": true, + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "safe-buffer": { + "version": "5.2.1", + "bundled": true, + "dev": true + }, + "safer-buffer": { + "version": "2.1.2", + "bundled": true, + "dev": true + }, "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "version": "7.3.5", + "bundled": true, + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "set-blocking": { + "version": "2.0.0", + "bundled": true, "dev": true - } - } - }, - "jsprim": { - "version": "1.4.2", - "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.2.tgz", - "integrity": "sha512-P2bSOMAc/ciLz6DzgjVlGJP9+BrJWu5UDGK70C2iweC5QBIeFf0ZXRvGjEj2uYgrY2MkAAhsSWHDWlFtEroZWw==", - "dev": true, - "requires": { - "assert-plus": "1.0.0", - "extsprintf": "1.3.0", - "json-schema": "0.4.0", - "verror": "1.10.0" - } - }, - "jwa": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.1.tgz", - "integrity": "sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA==", - "dev": true, - "requires": { - "buffer-equal-constant-time": "1.0.1", - "ecdsa-sig-formatter": "1.0.11", - "safe-buffer": "^5.0.1" - } - }, - "jws": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz", - "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==", - "dev": true, - "requires": { - "jwa": "^1.4.1", - "safe-buffer": "^5.0.1" - } - }, - "keypress": { - "version": "0.2.1", - "resolved": "https://registry.npmjs.org/keypress/-/keypress-0.2.1.tgz", - "integrity": "sha1-HoBFQlABjbrUw/6USX1uZ7YmnHc=", - "dev": true - }, - "klaw": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/klaw/-/klaw-3.0.0.tgz", - "integrity": "sha512-0Fo5oir+O9jnXu5EefYbVK+mHMBeEVEy2cmctR1O1NECcCkPRreJKrS6Qt/j3KC2C148Dfo9i3pCmCMsdqGr0g==", - "dev": true, - "requires": { - "graceful-fs": "^4.1.9" - } - }, - "kleur": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz", - "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w==", - "dev": true - }, - "lazystream": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/lazystream/-/lazystream-1.0.1.tgz", - "integrity": "sha512-b94GiNHQNy6JNTrt5w6zNyffMrNkXZb3KTkCZJb2V1xaEGCk093vkZ2jk3tpaeP33/OiXC+WvK9AxUebnf5nbw==", - "dev": true, - "requires": { - "readable-stream": "^2.0.5" - } - }, - "leven": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", - "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", - "dev": true - }, - "levn": { - "version": "0.3.0", - "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", - "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", - "dev": true, - "requires": { - "prelude-ls": "~1.1.2", - "type-check": "~0.3.2" - } - }, - "lilconfig": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-2.0.4.tgz", - "integrity": "sha512-bfTIN7lEsiooCocSISTWXkiWJkRqtL9wYtYy+8EK3Y41qh3mpwPU0ycTOgjdY9ErwXCc8QyrQp82bdL0Xkm9yA==", - "dev": true - }, - "line-column": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/line-column/-/line-column-1.0.2.tgz", - "integrity": "sha1-0lryk2tvSEkXKzEuR5LR2Ye8NKI=", - "dev": true, - "requires": { - "isarray": "^1.0.0", - "isobject": "^2.0.0" - }, - "dependencies": { - "isobject": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", - "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + }, + "signal-exit": { + "version": "3.0.3", + "bundled": true, + "dev": true + }, + "smart-buffer": { + "version": "4.2.0", + "bundled": true, + "dev": true + }, + "socks": { + "version": "2.6.1", + "bundled": true, "dev": true, "requires": { - "isarray": "1.0.0" + "ip": "^1.1.5", + "smart-buffer": "^4.1.0" } - } - } - }, - "linkify-it": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/linkify-it/-/linkify-it-2.2.0.tgz", - "integrity": "sha512-GnAl/knGn+i1U/wjBz3akz2stz+HrHLsxMwHQGofCDfPvlf+gDKN58UtfmUquTY4/MXeE2x7k19KQmeoZi94Iw==", - "dev": true, - "requires": { - "uc.micro": "^1.0.1" - } - }, - "lint-staged": { - "version": "12.2.1", - "resolved": "https://registry.npmjs.org/lint-staged/-/lint-staged-12.2.1.tgz", - "integrity": "sha512-VCVcA9C2Vt5HHxSR4EZVZFJcQRJH984CGBeY+cJ/xed4mBd+JidbM/xbKcCq5ASaygAV0iITtdsCTnID7h/1OQ==", - "dev": true, - "requires": { - "cli-truncate": "^3.1.0", - "colorette": "^2.0.16", - "commander": "^8.3.0", - "debug": "^4.3.3", - "execa": "^5.1.1", - "lilconfig": "2.0.4", - "listr2": "^3.13.5", - "micromatch": "^4.0.4", - "normalize-path": "^3.0.0", - "object-inspect": "^1.11.1", - "string-argv": "^0.3.1", - "supports-color": "^9.2.1", - "yaml": "^1.10.2" - }, - "dependencies": { - "commander": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/commander/-/commander-8.3.0.tgz", - "integrity": "sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww==", + }, + "socks-proxy-agent": { + "version": "6.1.0", + "bundled": true, + "dev": true, + "requires": { + "agent-base": "^6.0.2", + "debug": "^4.3.1", + "socks": "^2.6.1" + } + }, + "spdx-correct": { + "version": "3.1.1", + "bundled": true, + "dev": true, + "requires": { + "spdx-expression-parse": "^3.0.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-exceptions": { + "version": "2.3.0", + "bundled": true, "dev": true }, - "supports-color": { - "version": "9.2.1", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-9.2.1.tgz", - "integrity": "sha512-Obv7ycoCTG51N7y175StI9BlAXrmgZrFhZOb0/PyjHBher/NmsdBgbbQ1Inhq+gIhz6+7Gb+jWF2Vqi7Mf1xnQ==", + "spdx-expression-parse": { + "version": "3.0.1", + "bundled": true, + "dev": true, + "requires": { + "spdx-exceptions": "^2.1.0", + "spdx-license-ids": "^3.0.0" + } + }, + "spdx-license-ids": { + "version": "3.0.10", + "bundled": true, "dev": true - } - } - }, - "listr2": { - "version": "3.14.0", - "resolved": "https://registry.npmjs.org/listr2/-/listr2-3.14.0.tgz", - "integrity": "sha512-TyWI8G99GX9GjE54cJ+RrNMcIFBfwMPxc3XTFiAYGN4s10hWROGtOg7+O6u6LE3mNkyld7RSLE6nrKBvTfcs3g==", - "dev": true, - "requires": { - "cli-truncate": "^2.1.0", - "colorette": "^2.0.16", - "log-update": "^4.0.0", - "p-map": "^4.0.0", - "rfdc": "^1.3.0", - "rxjs": "^7.5.1", - "through": "^2.3.8", - "wrap-ansi": "^7.0.0" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "sshpk": { + "version": "1.16.1", + "bundled": true, + "dev": true, + "requires": { + "asn1": "~0.2.3", + "assert-plus": "^1.0.0", + "bcrypt-pbkdf": "^1.0.0", + "dashdash": "^1.12.0", + "ecc-jsbn": "~0.1.1", + "getpass": "^0.1.1", + "jsbn": "~0.1.0", + "safer-buffer": "^2.0.2", + "tweetnacl": "~0.14.0" + } + }, + "ssri": { + "version": "8.0.1", + "bundled": true, + "dev": true, + "requires": { + "minipass": "^3.1.1" + } + }, + "string-width": { + "version": "2.1.1", + "bundled": true, + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "bundled": true, + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "bundled": true, + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "string_decoder": { + "version": "1.3.0", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "safe-buffer": "~5.2.0" } }, - "astral-regex": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", - "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", + "stringify-package": { + "version": "1.0.1", + "bundled": true, "dev": true }, - "cli-truncate": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/cli-truncate/-/cli-truncate-2.1.0.tgz", - "integrity": "sha512-n8fOixwDD6b/ObinzTrp1ZKFzbgvKZvuz/TvejnLn1aQfC6r52XEx85FmuC+3HI+JM7coBRXUvNqEU2PHVrHpg==", + "strip-ansi": { + "version": "3.0.1", + "bundled": true, "dev": true, "requires": { - "slice-ansi": "^3.0.0", - "string-width": "^4.2.0" + "ansi-regex": "^2.0.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "supports-color": { + "version": "7.2.0", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "has-flag": "^4.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true - }, - "rxjs": { - "version": "7.5.2", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-7.5.2.tgz", - "integrity": "sha512-PwDt186XaL3QN5qXj/H9DGyHhP3/RYYgZZwqBv9Tv8rsAaiwFH1IsJJlcgD37J7UW5a6O67qX0KWKS3/pu0m4w==", + "tar": { + "version": "6.1.11", + "bundled": true, "dev": true, "requires": { - "tslib": "^2.1.0" + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "minipass": "^3.0.0", + "minizlib": "^2.1.1", + "mkdirp": "^1.0.3", + "yallist": "^4.0.0" } }, - "slice-ansi": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-3.0.0.tgz", - "integrity": "sha512-pSyv7bSTC7ig9Dcgbw9AuRNUb5k5V6oDudjZoMBSr13qpLBG7tB+zgCkARjq7xIUgdz5P1Qe8u+rSGdouOOIyQ==", + "text-table": { + "version": "0.2.0", + "bundled": true, + "dev": true + }, + "tiny-relative-date": { + "version": "1.3.0", + "bundled": true, + "dev": true + }, + "treeverse": { + "version": "1.0.4", + "bundled": true, + "dev": true + }, + "tunnel-agent": { + "version": "0.6.0", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.0.0", - "astral-regex": "^2.0.0", - "is-fullwidth-code-point": "^3.0.0" + "safe-buffer": "^5.0.1" } - } - } - }, - "load-json-file": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-2.0.0.tgz", - "integrity": "sha1-eUfkIUmvgNaWy/eXvKq8/h/inKg=", - "dev": true, - "requires": { - "graceful-fs": "^4.1.2", - "parse-json": "^2.2.0", - "pify": "^2.0.0", - "strip-bom": "^3.0.0" - }, - "dependencies": { - "strip-bom": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", - "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + }, + "tweetnacl": { + "version": "0.14.5", + "bundled": true, "dev": true - } - } - }, - "locate-path": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", - "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", - "dev": true, - "requires": { - "p-locate": "^5.0.0" - } - }, - "lodash": { - "version": "4.17.21", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", - "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", - "dev": true - }, - "lodash._reinterpolate": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/lodash._reinterpolate/-/lodash._reinterpolate-3.0.0.tgz", - "integrity": "sha1-DM8tiRZq8Ds2Y8eWU4t1rG4RTZ0=", - "dev": true - }, - "lodash.camelcase": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz", - "integrity": "sha1-soqmKIorn8ZRA1x3EfZathkDMaY=", - "dev": true - }, - "lodash.defaults": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", - "integrity": "sha1-0JF4cW/+pN3p5ft7N/bwgCJ0WAw=", - "dev": true - }, - "lodash.difference": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.difference/-/lodash.difference-4.5.0.tgz", - "integrity": "sha1-nMtOUF1Ia5FlE0V3KIWi3yf9AXw=", - "dev": true - }, - "lodash.flatten": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/lodash.flatten/-/lodash.flatten-4.4.0.tgz", - "integrity": "sha1-8xwiIlqWMtK7+OSt2+8kCqdlph8=", - "dev": true - }, - "lodash.get": { - "version": "4.4.2", - "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz", - "integrity": "sha1-LRd/ZS+jHpObRDjVNBSZ36OCXpk=", - "dev": true - }, - "lodash.includes": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz", - "integrity": "sha1-YLuYqHy5I8aMoeUTJUgzFISfVT8=", - "dev": true - }, - "lodash.isboolean": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz", - "integrity": "sha1-bC4XHbKiV82WgC/UOwGyDV9YcPY=", - "dev": true - }, - "lodash.isinteger": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz", - "integrity": "sha1-YZwK89A/iwTDH1iChAt3sRzWg0M=", - "dev": true - }, - "lodash.isnumber": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz", - "integrity": "sha1-POdoEMWSjQM1IwGsKHMX8RwLH/w=", - "dev": true - }, - "lodash.isplainobject": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz", - "integrity": "sha1-fFJqUtibRcRcxpC4gWO+BJf1UMs=", - "dev": true - }, - "lodash.isstring": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz", - "integrity": "sha1-1SfftUVuynzJu5XV2ur4i6VKVFE=", - "dev": true - }, - "lodash.memoize": { - "version": "4.1.2", - "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", - "integrity": "sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4=", - "dev": true - }, - "lodash.merge": { - "version": "4.6.2", - "resolved": "https://registry.npmjs.org/lodash.merge/-/lodash.merge-4.6.2.tgz", - "integrity": "sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ==", - "dev": true - }, - "lodash.omit": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.omit/-/lodash.omit-4.5.0.tgz", - "integrity": "sha1-brGa5aHuHdnfC5aeZs4Lf6MLXmA=", - "dev": true - }, - "lodash.once": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz", - "integrity": "sha1-DdOXEhPHxW34gJd9UEyI+0cal6w=", - "dev": true - }, - "lodash.padend": { - "version": "4.6.1", - "resolved": "https://registry.npmjs.org/lodash.padend/-/lodash.padend-4.6.1.tgz", - "integrity": "sha1-U8y6BH0G4VjTEfRdpiX05J5vFm4=", - "dev": true - }, - "lodash.pick": { - "version": "4.4.0", - "resolved": "https://registry.npmjs.org/lodash.pick/-/lodash.pick-4.4.0.tgz", - "integrity": "sha1-UvBWEP/53tQiYRRB7R/BI6AwAbM=", - "dev": true - }, - "lodash.set": { - "version": "4.3.2", - "resolved": "https://registry.npmjs.org/lodash.set/-/lodash.set-4.3.2.tgz", - "integrity": "sha1-2HV7HagH3eJIFrDWqEvqGnYjCyM=", - "dev": true - }, - "lodash.template": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.template/-/lodash.template-4.5.0.tgz", - "integrity": "sha512-84vYFxIkmidUiFxidA/KjjH9pAycqW+h980j7Fuz5qxRtO9pgB7MDFTdys1N7A5mcucRiDyEq4fusljItR1T/A==", - "dev": true, - "requires": { - "lodash._reinterpolate": "^3.0.0", - "lodash.templatesettings": "^4.0.0" - } - }, - "lodash.templatesettings": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/lodash.templatesettings/-/lodash.templatesettings-4.2.0.tgz", - "integrity": "sha512-stgLz+i3Aa9mZgnjr/O+v9ruKZsPsndy7qPZOchbqk2cnTU1ZaldKK+v7m54WoKIyxiuMZTKT2H81F8BeAc3ZQ==", - "dev": true, - "requires": { - "lodash._reinterpolate": "^3.0.0" - } - }, - "lodash.union": { - "version": "4.6.0", - "resolved": "https://registry.npmjs.org/lodash.union/-/lodash.union-4.6.0.tgz", - "integrity": "sha1-SLtQiECfFvGCFmZkHETdGqrjzYg=", - "dev": true - }, - "log-update": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/log-update/-/log-update-4.0.0.tgz", - "integrity": "sha512-9fkkDevMefjg0mmzWFBW8YkFP91OrizzkW3diF7CpG+S2EYdy4+TVfGwz1zeF8x7hCx1ovSPTOE9Ngib74qqUg==", - "dev": true, - "requires": { - "ansi-escapes": "^4.3.0", - "cli-cursor": "^3.1.0", - "slice-ansi": "^4.0.0", - "wrap-ansi": "^6.2.0" - }, - "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + }, + "typedarray-to-buffer": { + "version": "3.1.5", + "bundled": true, + "dev": true, + "requires": { + "is-typedarray": "^1.0.0" + } + }, + "unique-filename": { + "version": "1.1.1", + "bundled": true, "dev": true, "requires": { - "color-convert": "^2.0.1" + "unique-slug": "^2.0.0" } }, - "astral-regex": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/astral-regex/-/astral-regex-2.0.0.tgz", - "integrity": "sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ==", - "dev": true + "unique-slug": { + "version": "2.0.2", + "bundled": true, + "dev": true, + "requires": { + "imurmurhash": "^0.1.4" + } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "uri-js": { + "version": "4.4.1", + "bundled": true, "dev": true, "requires": { - "color-name": "~1.1.4" + "punycode": "^2.1.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "util-deprecate": { + "version": "1.0.2", + "bundled": true, "dev": true }, - "slice-ansi": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/slice-ansi/-/slice-ansi-4.0.0.tgz", - "integrity": "sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ==", + "uuid": { + "version": "3.4.0", + "bundled": true, + "dev": true + }, + "validate-npm-package-license": { + "version": "3.0.4", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.0.0", - "astral-regex": "^2.0.0", - "is-fullwidth-code-point": "^3.0.0" + "spdx-correct": "^3.0.0", + "spdx-expression-parse": "^3.0.0" } }, - "wrap-ansi": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", - "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "validate-npm-package-name": { + "version": "3.0.0", + "bundled": true, "dev": true, "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" + "builtins": "^1.0.3" } - } - } - }, - "lru-cache": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", - "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", - "dev": true, - "requires": { - "yallist": "^4.0.0" - } - }, - "make-dir": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", - "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", - "dev": true, - "requires": { - "semver": "^6.0.0" - } - }, - "make-error": { - "version": "1.3.6", - "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", - "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", - "dev": true - }, - "makeerror": { - "version": "1.0.12", - "resolved": "https://registry.npmjs.org/makeerror/-/makeerror-1.0.12.tgz", - "integrity": "sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==", - "dev": true, - "requires": { - "tmpl": "1.0.5" - } - }, - "markdown-it": { - "version": "12.3.2", - "resolved": "https://registry.npmjs.org/markdown-it/-/markdown-it-12.3.2.tgz", - "integrity": "sha512-TchMembfxfNVpHkbtriWltGWc+m3xszaRD0CZup7GFFhzIgQqxIfn3eGj1yZpfuflzPvfkt611B2Q/Bsk1YnGg==", - "requires": { - "argparse": "^2.0.1", - "entities": "~2.1.0", - "linkify-it": "^3.0.1", - "mdurl": "^1.0.1", - "uc.micro": "^1.0.5" - }, - "dependencies": { - "argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==" - }, - "entities": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/entities/-/entities-2.1.0.tgz", - "integrity": "sha512-hCx1oky9PFrJ611mf0ifBLBRW8lUUVRlFolb5gWRfIELabBlbp9xZvrqZLZAs+NxFnbfQoeGd8wDkygjg7U85w==" }, - "linkify-it": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/linkify-it/-/linkify-it-3.0.3.tgz", - "integrity": "sha512-ynTsyrFSdE5oZ/O9GEf00kPngmOfVwazR5GKDq6EYfhlpFug3J2zybX56a2PRRpc9P+FuSoGNAwjlbDs9jJBPQ==", + "verror": { + "version": "1.10.0", + "bundled": true, + "dev": true, "requires": { - "uc.micro": "^1.0.1" + "assert-plus": "^1.0.0", + "core-util-is": "1.0.2", + "extsprintf": "^1.2.0" } - } - } - }, - "markdown-it-anchor": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/markdown-it-anchor/-/markdown-it-anchor-5.3.0.tgz", - "integrity": "sha512-/V1MnLL/rgJ3jkMWo84UR+K+jF1cxNG1a+KwqeXqTIJ+jtA8aWSHuigx8lTzauiIjBDbwF3NcWQMotd0Dm39jA==", - "dev": true - }, - "marked": { - "version": "4.0.10", - "resolved": "https://registry.npmjs.org/marked/-/marked-4.0.10.tgz", - "integrity": "sha512-+QvuFj0nGgO970fySghXGmuw+Fd0gD2x3+MqCWLIPf5oxdv1Ka6b2q+z9RP01P/IaKPMEramy+7cNy/Lw8c3hw==" - }, - "mdurl": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/mdurl/-/mdurl-1.0.1.tgz", - "integrity": "sha1-/oWy7HWlkDfyrf7BAP1sYBdhFS4=" - }, - "merge-stream": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", - "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", - "dev": true - }, - "merge2": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", - "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", - "dev": true - }, - "micromatch": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz", - "integrity": "sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg==", - "dev": true, - "requires": { - "braces": "^3.0.1", - "picomatch": "^2.2.3" - } - }, - "mime-db": { - "version": "1.51.0", - "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.51.0.tgz", - "integrity": "sha512-5y8A56jg7XVQx2mbv1lu49NR4dokRnhZYTtL+KGfaa27uq4pSTXkwQkFJl4pkRMyNFz/EtYDSkiiEHx3F7UN6g==", - "dev": true - }, - "mime-types": { - "version": "2.1.34", - "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.34.tgz", - "integrity": "sha512-6cP692WwGIs9XXdOO4++N+7qjqv0rqxxVvJ3VHPh/Sc9mVZcQP+ZGhkKiTvWMQRr2tbHkJP/Yn7Y0npb3ZBs4A==", - "dev": true, - "requires": { - "mime-db": "1.51.0" - } - }, - "mimic-fn": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", - "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", - "dev": true - }, - "minimatch": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", - "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", - "requires": { - "brace-expansion": "^1.1.7" - } - }, - "minimist": { - "version": "1.2.5", - "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", - "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", - "dev": true - }, - "minipass": { - "version": "2.9.0", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz", - "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", - "dev": true, - "requires": { - "safe-buffer": "^5.1.2", - "yallist": "^3.0.0" - }, - "dependencies": { - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + }, + "walk-up-path": { + "version": "1.0.0", + "bundled": true, "dev": true - } - } - }, - "minizlib": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz", - "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", - "dev": true, - "requires": { - "minipass": "^2.9.0" - } - }, - "mkdirp": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", - "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", - "dev": true - }, - "mkdirp2": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/mkdirp2/-/mkdirp2-1.0.5.tgz", - "integrity": "sha512-xOE9xbICroUDmG1ye2h4bZ8WBie9EGmACaco8K8cx6RlkJJrxGIqjGqztAI+NMhexXBcdGbSEzI6N3EJPevxZw==", - "dev": true - }, - "mock-stdin": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/mock-stdin/-/mock-stdin-1.0.0.tgz", - "integrity": "sha512-tukRdb9Beu27t6dN+XztSRHq9J0B/CoAOySGzHfn8UTfmqipA5yNT/sDUEyYdAV3Hpka6Wx6kOMxuObdOex60Q==", - "dev": true - }, - "ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, - "multistream": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/multistream/-/multistream-2.1.1.tgz", - "integrity": "sha512-xasv76hl6nr1dEy3lPvy7Ej7K/Lx3O/FCvwge8PeVJpciPPoNCbaANcNiBug3IpdvTveZUcAV0DJzdnUDMesNQ==", - "dev": true, - "requires": { - "inherits": "^2.0.1", - "readable-stream": "^2.0.5" - } - }, - "mustache": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/mustache/-/mustache-4.2.0.tgz", - "integrity": "sha512-71ippSywq5Yb7/tVYyGbkBggbU8H3u5Rz56fH60jGFgr8uHwxs+aSKeqmluIVzM0m0kB7xQjKS6qPfd0b2ZoqQ==", - "dev": true - }, - "mute-stream": { - "version": "0.0.8", - "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", - "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", - "dev": true - }, - "mv": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/mv/-/mv-2.1.1.tgz", - "integrity": "sha1-rmzg1vbV4KT32JN5jQPB6pVZtqI=", - "dev": true, - "optional": true, - "requires": { - "mkdirp": "~0.5.1", - "ncp": "~2.0.0", - "rimraf": "~2.4.0" - }, - "dependencies": { - "glob": { - "version": "6.0.4", - "resolved": "https://registry.npmjs.org/glob/-/glob-6.0.4.tgz", - "integrity": "sha1-DwiGD2oVUSey+t1PnOJLGqtuTSI=", + }, + "wcwidth": { + "version": "1.0.1", + "bundled": true, + "dev": true, + "requires": { + "defaults": "^1.0.3" + } + }, + "which": { + "version": "2.0.2", + "bundled": true, "dev": true, - "optional": true, "requires": { - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "2 || 3", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" + "isexe": "^2.0.0" } }, - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + "wide-align": { + "version": "1.1.3", + "bundled": true, "dev": true, - "optional": true, "requires": { - "minimist": "^1.2.5" + "string-width": "^1.0.2 || 2" } }, - "rimraf": { - "version": "2.4.5", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.4.5.tgz", - "integrity": "sha1-7nEM5dk6j9uFb7Xqj/Di11k0sto=", + "wrappy": { + "version": "1.0.2", + "bundled": true, + "dev": true + }, + "write-file-atomic": { + "version": "3.0.3", + "bundled": true, "dev": true, - "optional": true, "requires": { - "glob": "^6.0.1" + "imurmurhash": "^0.1.4", + "is-typedarray": "^1.0.0", + "signal-exit": "^3.0.2", + "typedarray-to-buffer": "^3.1.5" } + }, + "yallist": { + "version": "4.0.0", + "bundled": true, + "dev": true } } }, - "nan": { - "version": "2.15.0", - "resolved": "https://registry.npmjs.org/nan/-/nan-2.15.0.tgz", - "integrity": "sha512-8ZtvEnA2c5aYCZYd1cvgdnU6cqwixRoYg70xPLWUws5ORTa/lnw+u4amixRS/Ac5U5mQVgp9pnlSUnbNWFaWZQ==", - "dev": true, - "optional": true - }, - "nanoid": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.2.0.tgz", - "integrity": "sha512-fmsZYa9lpn69Ad5eDn7FMcnnSR+8R34W9qJEijxYhTbfOWzr22n1QxCMzXLK+ODyW2973V3Fux959iQoUxzUIA==" - }, - "natural-compare": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz", - "integrity": "sha1-Sr6/7tdUHywnrPspvbvRXI1bpPc=", - "dev": true - }, - "ncp": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ncp/-/ncp-2.0.0.tgz", - "integrity": "sha1-GVoh1sRuNh0vsSgbo4uR6d9727M=", - "dev": true, - "optional": true - }, - "neo-async": { - "version": "2.6.2", - "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", - "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", - "dev": true - }, - "nice-try": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", - "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", - "dev": true - }, - "nock": { - "version": "13.2.1", - "resolved": "https://registry.npmjs.org/nock/-/nock-13.2.1.tgz", - "integrity": "sha512-CoHAabbqq/xZEknubuyQMjq6Lfi5b7RtK6SoNK6m40lebGp3yiMagWtIoYaw2s9sISD7wPuCfwFpivVHX/35RA==", - "dev": true, - "requires": { - "debug": "^4.1.0", - "json-stringify-safe": "^5.0.1", - "lodash.set": "^4.3.2", - "propagate": "^2.0.0" - } - }, - "node-int64": { - "version": "0.4.0", - "resolved": "https://registry.npmjs.org/node-int64/-/node-int64-0.4.0.tgz", - "integrity": "sha1-h6kGXNs1XTGC2PlM4RGIuCXGijs=", - "dev": true - }, - "node-releases": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.1.tgz", - "integrity": "sha512-CqyzN6z7Q6aMeF/ktcMVTzhAHCEpf8SOarwpzpf8pNBY2k5/oM34UHldUwp8VKI7uxct2HxSRdJjBaZeESzcxA==", - "dev": true - }, - "node-stream-zip": { - "version": "1.13.2", - "resolved": "https://registry.npmjs.org/node-stream-zip/-/node-stream-zip-1.13.2.tgz", - "integrity": "sha512-159NUO3quDIRVSVOQiQ/0VG8NFrGvVyUa+MUUleiOys3NLtrGtNkodqYYTRHU/kiXu/ygHZcNy/tZ4NUCcDUmQ==", - "dev": true - }, - "normalize-package-data": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", - "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "npm-api": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/npm-api/-/npm-api-1.0.1.tgz", + "integrity": "sha512-4sITrrzEbPcr0aNV28QyOmgn6C9yKiF8k92jn4buYAK8wmA5xo1qL3II5/gT1r7wxbXBflSduZ2K3FbtOrtGkA==", "dev": true, "requires": { - "hosted-git-info": "^2.1.4", - "resolve": "^1.10.0", - "semver": "2 || 3 || 4 || 5", - "validate-npm-package-license": "^3.0.1" - }, - "dependencies": { - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } + "JSONStream": "^1.3.5", + "clone-deep": "^4.0.1", + "download-stats": "^0.3.4", + "moment": "^2.24.0", + "node-fetch": "^2.6.0", + "paged-request": "^2.0.1" } }, - "normalize-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", - "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true - }, "npm-normalize-package-bin": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz", @@ -8367,8 +17592,22 @@ "dev": true, "requires": { "path-key": "^3.0.0" + }, + "dependencies": { + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + } } }, + "number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", + "dev": true + }, "nwsapi": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.0.tgz", @@ -8381,6 +17620,43 @@ "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", "dev": true }, + "object-assign": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", + "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", + "dev": true + }, + "object-copy": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", + "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", + "dev": true, + "requires": { + "copy-descriptor": "^0.1.0", + "define-property": "^0.2.5", + "kind-of": "^3.0.3" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, "object-get": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/object-get/-/object-get-2.1.1.tgz", @@ -8405,6 +17681,29 @@ "integrity": "sha512-6FuKFQ39cOID+BMZ3QaphcC8Y4cw6LXBLyIgPU+OhIYwviJamPAn+4mITapnSBQrejB+NNp+FMskhD8Cq+Ys3w==", "dev": true }, + "object-treeify": { + "version": "1.1.33", + "resolved": "https://registry.npmjs.org/object-treeify/-/object-treeify-1.1.33.tgz", + "integrity": "sha512-EFVjAYfzWqWsBMRHPMAXLCDIJnpMhdWAqR7xG6M6a2cs6PMFpl/+Z20w9zDW4vkxOFfddegBKq9Rehd0bxWE7A==", + "dev": true + }, + "object-visit": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", + "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", + "dev": true, + "requires": { + "isobject": "^3.0.0" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, "object.assign": { "version": "4.1.2", "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", @@ -8417,6 +17716,23 @@ "object-keys": "^1.1.1" } }, + "object.pick": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", + "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", + "dev": true, + "requires": { + "isobject": "^3.0.1" + }, + "dependencies": { + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, "object.values": { "version": "1.1.5", "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.5.tgz", @@ -8445,6 +17761,16 @@ "mimic-fn": "^2.1.0" } }, + "open": { + "version": "7.4.2", + "resolved": "https://registry.npmjs.org/open/-/open-7.4.2.tgz", + "integrity": "sha512-MVHddDVweXZF3awtlAS+6pgKLlm/JgxZ90+/NBurBoQctVOOB/zDdVjcyPzQ+0laDGbsWgrRkflI65sQeOgT9Q==", + "dev": true, + "requires": { + "is-docker": "^2.0.0", + "is-wsl": "^2.1.1" + } + }, "opn": { "version": "5.5.0", "resolved": "https://registry.npmjs.org/opn/-/opn-5.5.0.tgz", @@ -8462,6 +17788,12 @@ } } }, + "optional-js": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/optional-js/-/optional-js-1.3.1.tgz", + "integrity": "sha1-HW8knW2kaC5ps1u49bOrLhy12Wo=", + "dev": true + }, "optionator": { "version": "0.8.3", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", @@ -8482,6 +17814,24 @@ "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", "dev": true }, + "p-cancelable": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/p-cancelable/-/p-cancelable-2.1.1.tgz", + "integrity": "sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg==", + "dev": true + }, + "p-finally": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", + "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", + "dev": true + }, + "p-is-promise": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/p-is-promise/-/p-is-promise-1.1.0.tgz", + "integrity": "sha1-nJRWmJ6fZYgBewQ01WCXZ1w9oF4=", + "dev": true + }, "p-limit": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", @@ -8509,12 +17859,80 @@ "aggregate-error": "^3.0.0" } }, + "p-timeout": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/p-timeout/-/p-timeout-2.0.1.tgz", + "integrity": "sha512-88em58dDVB/KzPEx1X0N3LwFfYZPyDc4B6eF38M1rk9VTZMbxXXgjugz8mmwpS9Ox4BDZ+t6t3QP5+/gazweIA==", + "dev": true, + "requires": { + "p-finally": "^1.0.0" + } + }, "p-try": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", "dev": true }, + "pac-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/pac-proxy-agent/-/pac-proxy-agent-5.0.0.tgz", + "integrity": "sha512-CcFG3ZtnxO8McDigozwE3AqAw15zDvGH+OjXO4kzf7IkEKkQ4gxQ+3sdF50WmhQ4P/bVusXcqNE2S3XrNURwzQ==", + "dev": true, + "requires": { + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4", + "get-uri": "3", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "5", + "pac-resolver": "^5.0.0", + "raw-body": "^2.2.0", + "socks-proxy-agent": "5" + } + }, + "pac-resolver": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/pac-resolver/-/pac-resolver-5.0.0.tgz", + "integrity": "sha512-H+/A6KitiHNNW+bxBKREk2MCGSxljfqRX76NjummWEYIat7ldVXRU3dhRIE3iXZ0nvGBk6smv3nntxKkzRL8NA==", + "dev": true, + "requires": { + "degenerator": "^3.0.1", + "ip": "^1.1.5", + "netmask": "^2.0.1" + } + }, + "pad-component": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/pad-component/-/pad-component-0.0.1.tgz", + "integrity": "sha1-rR8izhvw/cDW3dkIrxfzUaQEuKw=", + "dev": true + }, + "paged-request": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/paged-request/-/paged-request-2.0.2.tgz", + "integrity": "sha512-NWrGqneZImDdcMU/7vMcAOo1bIi5h/pmpJqe7/jdsy85BA/s5MSaU/KlpxwW/IVPmIwBcq2uKPrBWWhEWhtxag==", + "dev": true, + "requires": { + "axios": "^0.21.1" + } + }, + "pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==", + "dev": true + }, + "param-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/param-case/-/param-case-3.0.4.tgz", + "integrity": "sha512-RXlj7zCYokReqWpOPH9oYivUzLYZ5vAPIfEmCTNViosC78F8F0H9y7T7gG2M39ymgutxF5gcFEsyZQSph9Bp3A==", + "dev": true, + "requires": { + "dot-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, "parent-module": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", @@ -8525,12 +17943,15 @@ } }, "parse-json": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-2.2.0.tgz", - "integrity": "sha1-9ID0BDTvgHQfhGkJn43qGPVaTck=", + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", "dev": true, "requires": { - "error-ex": "^1.2.0" + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" } }, "parse5": { @@ -8546,75 +17967,58 @@ "dev": true, "requires": { "@types/node": "^6.0.46" - } - }, - "password-prompt": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/password-prompt/-/password-prompt-1.1.2.tgz", - "integrity": "sha512-bpuBhROdrhuN3E7G/koAju0WjVw9/uQOG5Co5mokNj0MiOSBVZS1JTwM4zl55hu0WFmIEFvO9cU9sJQiBIYeIA==", - "dev": true, - "requires": { - "ansi-escapes": "^3.1.0", - "cross-spawn": "^6.0.5" }, "dependencies": { - "ansi-escapes": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-3.2.0.tgz", - "integrity": "sha512-cBhpre4ma+U0T1oM5fXg7Dy1Jw7zzwv7lt/GoCpr+hDQJoYnKVPLL4dCvSEFMmQurOQvSrwT7SL/DAlhBI97RQ==", - "dev": true - }, - "cross-spawn": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", - "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", - "dev": true, - "requires": { - "nice-try": "^1.0.4", - "path-key": "^2.0.1", - "semver": "^5.5.0", - "shebang-command": "^1.2.0", - "which": "^1.2.9" - } - }, - "path-key": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", - "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "shebang-command": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", - "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", - "dev": true, - "requires": { - "shebang-regex": "^1.0.0" - } - }, - "shebang-regex": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", - "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", + "@types/node": { + "version": "6.14.13", + "resolved": "https://registry.npmjs.org/@types/node/-/node-6.14.13.tgz", + "integrity": "sha512-J1F0XJ/9zxlZel5ZlbeSuHW2OpabrUAqpFuC2sm2I3by8sERQ8+KCjNKUcq8QHuzpGMWiJpo9ZxeHrqrP2KzQw==", "dev": true - }, - "which": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", - "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", - "dev": true, - "requires": { - "isexe": "^2.0.0" - } } } }, + "pascal-case": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/pascal-case/-/pascal-case-3.1.2.tgz", + "integrity": "sha512-uWlGT3YSnK9x3BQJaOdcZwrnV6hPpd8jFH1/ucpiLRPh/2zCVJKS19E4GvYHvaCcACn3foXZ0cLB9Wrx1KGe5g==", + "dev": true, + "requires": { + "no-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "pascalcase": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", + "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", + "dev": true + }, + "password-prompt": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/password-prompt/-/password-prompt-1.1.2.tgz", + "integrity": "sha512-bpuBhROdrhuN3E7G/koAju0WjVw9/uQOG5Co5mokNj0MiOSBVZS1JTwM4zl55hu0WFmIEFvO9cU9sJQiBIYeIA==", + "dev": true, + "requires": { + "ansi-escapes": "^3.1.0", + "cross-spawn": "^6.0.5" + } + }, + "path-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/path-case/-/path-case-3.0.4.tgz", + "integrity": "sha512-qO4qCFjXqVTrcbPt/hQfhTQ+VhFsqNKOPtytgNKkKxSoEp3XPUQ8ObFuePylOIok5gjn69ry8XiULxCwot3Wfg==", + "dev": true, + "requires": { + "dot-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "path-dirname": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", + "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", + "dev": true + }, "path-exists": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", @@ -8627,9 +18031,9 @@ "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=" }, "path-key": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", - "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", + "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", "dev": true }, "path-parse": { @@ -8656,9 +18060,15 @@ "dev": true }, "picomatch": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", - "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", + "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", + "dev": true + }, + "pidtree": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/pidtree/-/pidtree-0.5.0.tgz", + "integrity": "sha512-9nxspIM7OpZuhBxPg73Zvyq7j1QMPMPsGKTqRc2XOaFQauDvoNz9fM1Wdkjmeo7l9GXOZiRs97sPkuayl39wjA==", "dev": true }, "pify": { @@ -8668,9 +18078,9 @@ "dev": true }, "pirates": { - "version": "4.0.4", - "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.4.tgz", - "integrity": "sha512-ZIrVPH+A52Dw84R0L3/VS9Op04PuQ2SEoJL6bkshmiTic/HldyW9Tf7oH5mhJZBK7NmDx27vSMrYEXPXclpDKw==", + "version": "4.0.5", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.5.tgz", + "integrity": "sha512-8V9+HQPupnaXMA23c5hvl69zXvTwTzyAYasnkb0Tts4XvO4CliqONMOnvlq26rkhLC3nWDFBJf73LU1e1VZLaQ==", "dev": true }, "pkg-dir": { @@ -8721,6 +18131,12 @@ } } }, + "posix-character-classes": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", + "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", + "dev": true + }, "postcss": { "version": "8.3.11", "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.3.11.tgz", @@ -8733,9 +18149,9 @@ } }, "postcss-selector-parser": { - "version": "6.0.8", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.8.tgz", - "integrity": "sha512-D5PG53d209Z1Uhcc0qAZ5U3t5HagH3cxu+WLZ22jt3gLUpXM4eXXfiO14jiDWST3NNooX/E8wISfOhZ9eIjGTQ==", + "version": "6.0.10", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.10.tgz", + "integrity": "sha512-IQ7TZdoaqbT+LCpShg46jnZVlhWD2w6iQYAcYXfHARZ7X1t/UGhhceQDs5X0cGqKvYlHNOuv7Oa1xmb0oQuA3w==", "dev": true, "requires": { "cssesc": "^3.0.0", @@ -8754,10 +18170,16 @@ "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", "dev": true }, + "prepend-http": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", + "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", + "dev": true + }, "prettier": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.5.1.tgz", - "integrity": "sha512-vBZcPRUR5MZJwoyi3ZoyQlc1rXeEck8KgeC9AwwOn+exuxLxq5toTRDTSaVrXHxelDMHy9zlicw8u66yxoSUFg==", + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.6.2.tgz", + "integrity": "sha512-PkUpF+qoXTqhOeWL9fu7As8LXsIUZ1WYaJiY/a7McAQzxjk82OF0tibkFXVCDImZtWxbvojFjerkiLb0/q8mew==", "dev": true }, "prettier-plugin-apex": { @@ -8773,9 +18195,9 @@ }, "dependencies": { "yargs": { - "version": "17.3.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.3.1.tgz", - "integrity": "sha512-WUANQeVgjLbNsEmGk20f+nlHgOqzRFpiGWVaBrYGYIGANIIu3lWjoyi0fNlFmJkvfhCZ6BXINe7/W2O2bV4iaA==", + "version": "17.4.1", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.4.1.tgz", + "integrity": "sha512-WSZD9jgobAg3ZKuCQZSa3g9QOJeCCqLoLAykiWgmXnDo9EPnn4RPf5qVTtzgOx66o6/oqhcA5tHtJXpG8pMt3g==", "dev": true, "requires": { "cliui": "^7.0.2", @@ -8788,20 +18210,25 @@ } }, "yargs-parser": { - "version": "21.0.0", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.0.0.tgz", - "integrity": "sha512-z9kApYUOCwoeZ78rfRYYWdiU/iNL6mwwYlkkZfJoyMR1xps+NEBX5X7XmRpxkZHhXJ6+Ey00IwKxBBSW9FIjyA==", + "version": "21.0.1", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-21.0.1.tgz", + "integrity": "sha512-9BK1jFpLzJROCI5TzwZL/TU4gqjK5xiHV/RfWLOahrjAko/e4DJkRDZQXfvqAsiZzzYhgAzbgz6lg48jcm4GLg==", "dev": true } } }, + "pretty-bytes": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-5.6.0.tgz", + "integrity": "sha512-FFw039TmrBqFK8ma/7OL3sDz/VytdtJr044/QUJtH0wK9lb9jLq9tJyIxUwtQJHwar2BqtiA4iCWSwo9JLkzFg==", + "dev": true + }, "pretty-format": { - "version": "27.4.2", - "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.4.2.tgz", - "integrity": "sha512-p0wNtJ9oLuvgOQDEIZ9zQjZffK7KtyR6Si0jnXULIDwrlNF8Cuir3AZP0hHv0jmKuNN/edOnbMjnzd4uTcmWiw==", + "version": "27.5.1", + "resolved": "https://registry.npmjs.org/pretty-format/-/pretty-format-27.5.1.tgz", + "integrity": "sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ==", "dev": true, "requires": { - "@jest/types": "^27.4.2", "ansi-regex": "^5.0.1", "ansi-styles": "^5.0.0", "react-is": "^17.0.1" @@ -8815,12 +18242,6 @@ } } }, - "printj": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/printj/-/printj-1.3.1.tgz", - "integrity": "sha512-GA3TdL8szPK4AQ2YnOe/b+Y1jUFwmmGMMK/qbY7VcE3Z7FU8JstbKiKRzO6CIiAKPhTO8m01NoQ0V5f3jc4OGg==", - "dev": true - }, "process-nextick-args": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", @@ -8858,12 +18279,55 @@ "integrity": "sha512-vGrhOavPSTz4QVNuBNdcNXePNdNMaO1xj9yBeH1ScQPjk/rhg9sSlCXPhMkFuaNNW/syTvYqsnbIJxMBfRbbag==", "dev": true }, + "proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/proxy-agent/-/proxy-agent-5.0.0.tgz", + "integrity": "sha512-gkH7BkvLVkSfX9Dk27W6TyNOWWZWRilRfk1XxGNWOYJ2TuedAv1yFpCaU9QSBmBe716XOTNpYNOzhysyw8xn7g==", + "dev": true, + "requires": { + "agent-base": "^6.0.0", + "debug": "4", + "http-proxy-agent": "^4.0.0", + "https-proxy-agent": "^5.0.0", + "lru-cache": "^5.1.1", + "pac-proxy-agent": "^5.0.0", + "proxy-from-env": "^1.0.0", + "socks-proxy-agent": "^5.0.0" + }, + "dependencies": { + "lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "requires": { + "yallist": "^3.0.2" + } + } + } + }, + "proxy-from-env": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz", + "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==", + "dev": true + }, "psl": { "version": "1.8.0", "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==", "dev": true }, + "pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "dev": true, + "requires": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, "punycode": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", @@ -8880,23 +18344,90 @@ } }, "qs": { - "version": "6.5.2", - "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", - "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==", + "version": "6.5.3", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.3.tgz", + "integrity": "sha512-qxXIEh4pCGfHICj1mAJQ2/2XVZkjCDTcEgfoSQxc/fYivUZxTkk7L3bDBJSoNrEzXI17oUO5Dp07ktqE5KzczA==", "dev": true }, + "query-string": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/query-string/-/query-string-5.1.1.tgz", + "integrity": "sha512-gjWOsm2SoGlgLEdAGt7a6slVOk9mGiXmPFMqrEhLQ68rhQuBnpfs3+EmlvqKyxnCo9/PPlF+9MtY02S1aFg+Jw==", + "dev": true, + "requires": { + "decode-uri-component": "^0.2.0", + "object-assign": "^4.1.0", + "strict-uri-encode": "^1.0.0" + } + }, "queue-microtask": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", "dev": true }, + "quick-lru": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/quick-lru/-/quick-lru-5.1.1.tgz", + "integrity": "sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA==", + "dev": true + }, + "raw-body": { + "version": "2.5.1", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.5.1.tgz", + "integrity": "sha512-qqJBtEyVgS0ZmPGdCFPWJ3FreoqvG4MVQln/kCgF7Olq95IbOp0/BWyMwbdtn4VTvkM8Y7khCQ2Xgk/tcrCXig==", + "dev": true, + "requires": { + "bytes": "3.1.2", + "http-errors": "2.0.0", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + } + }, + "rc": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz", + "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==", + "dev": true, + "requires": { + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "dependencies": { + "strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true + } + } + }, "react-is": { "version": "17.0.2", "resolved": "https://registry.npmjs.org/react-is/-/react-is-17.0.2.tgz", "integrity": "sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w==", "dev": true }, + "read-chunk": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/read-chunk/-/read-chunk-3.2.0.tgz", + "integrity": "sha512-CEjy9LCzhmD7nUpJ1oVOE6s/hBkejlcJEgLQHVnQznOSilOPb+kpKktlLfFDK3/WP43+F80xkUTM2VOkYoSYvQ==", + "dev": true, + "requires": { + "pify": "^4.0.1", + "with-open-file": "^0.1.6" + }, + "dependencies": { + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + } + } + }, "read-installed": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/read-installed/-/read-installed-4.0.3.tgz", @@ -9016,18 +18547,14 @@ } }, "readable-stream": { - "version": "2.3.7", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", - "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", "dev": true, "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" } }, "readdir-glob": { @@ -9095,67 +18622,238 @@ "array-back": "^1.0.2", "typical": "^2.4.2" } - } - } - }, - "reduce-flatten": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/reduce-flatten/-/reduce-flatten-1.0.1.tgz", - "integrity": "sha1-JYx479FT3fk8tWEjf2EYTzaW4yc=", - "dev": true - }, - "reduce-unique": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/reduce-unique/-/reduce-unique-2.0.1.tgz", - "integrity": "sha512-x4jH/8L1eyZGR785WY+ePtyMNhycl1N2XOLxhCbzZFaqF4AXjLzqSxa2UHgJ2ZVR/HHyPOvl1L7xRnW8ye5MdA==", - "dev": true - }, - "reduce-without": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/reduce-without/-/reduce-without-1.0.1.tgz", - "integrity": "sha1-aK0OrRGFXJo31OglbBW7+Hly/Iw=", - "dev": true, - "requires": { - "test-value": "^2.0.0" - }, - "dependencies": { - "array-back": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/array-back/-/array-back-1.0.4.tgz", - "integrity": "sha1-ZEun8JX3/898Q7Xw3DnTwfA8Bjs=", + } + } + }, + "reduce-flatten": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/reduce-flatten/-/reduce-flatten-1.0.1.tgz", + "integrity": "sha1-JYx479FT3fk8tWEjf2EYTzaW4yc=", + "dev": true + }, + "reduce-unique": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/reduce-unique/-/reduce-unique-2.0.1.tgz", + "integrity": "sha512-x4jH/8L1eyZGR785WY+ePtyMNhycl1N2XOLxhCbzZFaqF4AXjLzqSxa2UHgJ2ZVR/HHyPOvl1L7xRnW8ye5MdA==", + "dev": true + }, + "reduce-without": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/reduce-without/-/reduce-without-1.0.1.tgz", + "integrity": "sha1-aK0OrRGFXJo31OglbBW7+Hly/Iw=", + "dev": true, + "requires": { + "test-value": "^2.0.0" + }, + "dependencies": { + "array-back": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/array-back/-/array-back-1.0.4.tgz", + "integrity": "sha1-ZEun8JX3/898Q7Xw3DnTwfA8Bjs=", + "dev": true, + "requires": { + "typical": "^2.6.0" + } + }, + "test-value": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/test-value/-/test-value-2.1.0.tgz", + "integrity": "sha1-Edpv9nDzRxpztiXKTz/c97t0gpE=", + "dev": true, + "requires": { + "array-back": "^1.0.3", + "typical": "^2.6.0" + } + } + } + }, + "reflect-metadata": { + "version": "0.1.13", + "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", + "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==", + "dev": true + }, + "regenerator-runtime": { + "version": "0.13.9", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.9.tgz", + "integrity": "sha512-p3VT+cOEgxFsRRA9X4lkI1E+k2/CtnKtU4gcxyaCUreilL/vqI6CdZ3wxVUx3UOUg+gnUOQQcRI7BmSI656MYA==", + "dev": true + }, + "regex-not": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", + "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", + "dev": true, + "requires": { + "extend-shallow": "^3.0.2", + "safe-regex": "^1.1.0" + } + }, + "regexp-to-ast": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/regexp-to-ast/-/regexp-to-ast-0.5.0.tgz", + "integrity": "sha512-tlbJqcMHnPKI9zSrystikWKwHkBqu2a/Sgw01h3zFjvYrMxEDYHzzoMZnUrbIfpTFEsoRnnviOXNCzFiSc54Qw==", + "dev": true + }, + "regexpp": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", + "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", + "dev": true + }, + "remove-trailing-separator": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", + "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", + "dev": true + }, + "repeat-element": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.4.tgz", + "integrity": "sha512-LFiNfRcSu7KK3evMyYOuCzv3L10TW7yC1G2/+StMjK8Y6Vqd2MG7r/Qjw4ghtuCOjFvlnms/iMmLqpvW/ES/WQ==", + "dev": true + }, + "repeat-string": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", + "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", + "dev": true + }, + "replace": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/replace/-/replace-1.2.1.tgz", + "integrity": "sha512-KZCBe/tPanwBlbjSMQby4l+zjSiFi3CLEP/6VLClnRYgJ46DZ5u9tmA6ceWeFS8coaUnU4ZdGNb/puUGMHNSRg==", + "dev": true, + "requires": { + "chalk": "2.4.2", + "minimatch": "3.0.4", + "yargs": "^15.3.1" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "cliui": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", + "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", + "dev": true, + "requires": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^6.2.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "requires": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + } + }, + "locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "requires": { + "p-locate": "^4.1.0" + } + }, + "p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "requires": { + "p-try": "^2.0.0" + } + }, + "p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "requires": { + "p-limit": "^2.2.0" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + } + }, + "y18n": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz", + "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==", + "dev": true + }, + "yargs": { + "version": "15.4.1", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", + "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", "dev": true, "requires": { - "typical": "^2.6.0" + "cliui": "^6.0.0", + "decamelize": "^1.2.0", + "find-up": "^4.1.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^4.2.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^18.1.2" } }, - "test-value": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/test-value/-/test-value-2.1.0.tgz", - "integrity": "sha1-Edpv9nDzRxpztiXKTz/c97t0gpE=", + "yargs-parser": { + "version": "18.1.3", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", + "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", "dev": true, "requires": { - "array-back": "^1.0.3", - "typical": "^2.6.0" + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" } } } }, - "reflect-metadata": { - "version": "0.1.13", - "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", - "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==", - "dev": true - }, - "regexp-to-ast": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/regexp-to-ast/-/regexp-to-ast-0.5.0.tgz", - "integrity": "sha512-tlbJqcMHnPKI9zSrystikWKwHkBqu2a/Sgw01h3zFjvYrMxEDYHzzoMZnUrbIfpTFEsoRnnviOXNCzFiSc54Qw==", - "dev": true - }, - "regexpp": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/regexpp/-/regexpp-3.2.0.tgz", - "integrity": "sha512-pq2bWo9mVD43nbts2wGv17XLiNLya+GklZ8kaDLV2Z08gDCsGpnKn9BFMepvWuHCbyVvY7J5o5+BVvoQbmlJLg==", + "replace-ext": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/replace-ext/-/replace-ext-1.0.1.tgz", + "integrity": "sha512-yD5BHCe7quCgBph4rMQ+0KkIRKwWCrHDOX1p1Gp6HwjPM5kVoCdKGNhN7ydqqsX6lJEnQDKZ/tFMiEdQ1dvPEw==", "dev": true }, "request": { @@ -9186,17 +18884,6 @@ "uuid": "^3.3.2" }, "dependencies": { - "form-data": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", - "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", - "dev": true, - "requires": { - "asynckit": "^0.4.0", - "combined-stream": "^1.0.6", - "mime-types": "^2.1.12" - } - }, "tough-cookie": { "version": "2.5.0", "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", @@ -9215,6 +18902,12 @@ "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", "dev": true }, + "require-main-filename": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", + "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", + "dev": true + }, "requizzle": { "version": "0.2.3", "resolved": "https://registry.npmjs.org/requizzle/-/requizzle-0.2.3.tgz", @@ -9233,6 +18926,12 @@ "path-parse": "^1.0.6" } }, + "resolve-alpn": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/resolve-alpn/-/resolve-alpn-1.2.1.tgz", + "integrity": "sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g==", + "dev": true + }, "resolve-cwd": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-3.0.0.tgz", @@ -9248,12 +18947,35 @@ "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", "dev": true }, + "resolve-url": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", + "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", + "dev": true + }, "resolve.exports": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/resolve.exports/-/resolve.exports-1.1.0.tgz", "integrity": "sha512-J1l+Zxxp4XK3LUDZ9m60LRJF/mAe4z6a4xyabPHk7pvK5t35dACV32iIjJDFeWZFfZlO29w6SZ67knR0tHzJtQ==", "dev": true }, + "responselike": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/responselike/-/responselike-2.0.0.tgz", + "integrity": "sha512-xH48u3FTB9VsZw7R+vvgaKeLKzT6jOogbQhEe/jewwnZgzPcnyWui2Av6JpoYZF/91uueC+lqhWqeURw5/qhCw==", + "dev": true, + "requires": { + "lowercase-keys": "^2.0.0" + }, + "dependencies": { + "lowercase-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/lowercase-keys/-/lowercase-keys-2.0.0.tgz", + "integrity": "sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA==", + "dev": true + } + } + }, "restore-cursor": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", @@ -9264,6 +18986,12 @@ "signal-exit": "^3.0.2" } }, + "ret": { + "version": "0.1.15", + "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", + "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", + "dev": true + }, "retire": { "version": "2.2.5", "resolved": "https://registry.npmjs.org/retire/-/retire-2.2.5.tgz", @@ -9298,12 +19026,27 @@ "dev": true }, "rimraf": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", - "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "version": "2.4.5", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.4.5.tgz", + "integrity": "sha1-7nEM5dk6j9uFb7Xqj/Di11k0sto=", "dev": true, "requires": { - "glob": "^7.1.3" + "glob": "^6.0.1" + }, + "dependencies": { + "glob": { + "version": "6.0.4", + "resolved": "https://registry.npmjs.org/glob/-/glob-6.0.4.tgz", + "integrity": "sha1-DwiGD2oVUSey+t1PnOJLGqtuTSI=", + "dev": true, + "requires": { + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "2 || 3", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + } } }, "run-async": { @@ -9338,25 +19081,334 @@ } } }, - "safe-buffer": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", - "dev": true - }, - "safe-json-stringify": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/safe-json-stringify/-/safe-json-stringify-1.2.0.tgz", - "integrity": "sha512-gH8eh2nZudPQO6TytOvbxnuhYBOvDBBLW52tz5q6X58lJcd/tkmqFR+5Z9adS8aJtURSXWThWy/xJtJwixErvg==", - "dev": true, - "optional": true - }, - "safer-buffer": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", - "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", - "dev": true - }, + "safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true + }, + "safe-json-stringify": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/safe-json-stringify/-/safe-json-stringify-1.2.0.tgz", + "integrity": "sha512-gH8eh2nZudPQO6TytOvbxnuhYBOvDBBLW52tz5q6X58lJcd/tkmqFR+5Z9adS8aJtURSXWThWy/xJtJwixErvg==", + "dev": true, + "optional": true + }, + "safe-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", + "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", + "dev": true, + "requires": { + "ret": "~0.1.10" + } + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", + "dev": true + }, + "salesforce-alm": { + "version": "54.1.0", + "resolved": "https://registry.npmjs.org/salesforce-alm/-/salesforce-alm-54.1.0.tgz", + "integrity": "sha512-orQcWk/TgQjrvHlzsFk2Pbm8fnh45eg765Q8tU1SvqAKZzyoLMkwwtOoP37wEEl7M6t3NyvvuFTnG6GvvOZflQ==", + "dev": true, + "requires": { + "@oclif/config": "^1.18.1", + "@oclif/errors": "^1.3.5", + "@salesforce/command": "^4.2.2", + "@salesforce/core": "^2.35.0", + "@salesforce/kit": "^1.5.13", + "@salesforce/source-deploy-retrieve": "^5.9.4", + "@salesforce/source-tracking": "^1.1.1", + "@salesforce/ts-types": "^1.5.17", + "adm-zip": "^0.5.9", + "ansi-styles": "^3.2.1", + "archiver": "^5.3.0", + "bluebird": "^3.5.5", + "bunyan-sfdx-no-dtrace": "^1.8.2", + "chalk": "^2.4.2", + "cli-ux": "^5.4.4", + "debug": "^3.2.6", + "fast-xml-parser": "^3.17.4", + "fs-extra": "^4.0.3", + "glob": "^7.1.6", + "heroku-cli-util": "^8.0.12", + "js2xmlparser": "^3.0.0", + "jsforce": "^1.11.0", + "klaw": "^2.1.1", + "lodash": "^4.17.21", + "mime": "^1.6.0", + "mkdirp": "^0.5.1", + "moment": "^2.24.0", + "optional-js": "^1.3.1", + "replace": "^1.2.1", + "request": "^2.88.0", + "strip-ansi": "^5.2.0", + "ts-retry-promise": "^0.6.0", + "xml2js": "0.4.19", + "xmldom-sfdx-encoding": "^0.1.30" + }, + "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "bunyan-sfdx-no-dtrace": { + "version": "1.8.2", + "resolved": "https://registry.npmjs.org/bunyan-sfdx-no-dtrace/-/bunyan-sfdx-no-dtrace-1.8.2.tgz", + "integrity": "sha1-EGgt4H/sF+0UUosWaZWx8Yu9T5o=", + "dev": true, + "requires": { + "dtrace-provider": "~0.6", + "moment": "^2.10.6", + "mv": "~2", + "safe-json-stringify": "~1" + } + }, + "cli-ux": { + "version": "5.6.7", + "resolved": "https://registry.npmjs.org/cli-ux/-/cli-ux-5.6.7.tgz", + "integrity": "sha512-dsKAurMNyFDnO6X1TiiRNiVbL90XReLKcvIq4H777NMqXGBxBws23ag8ubCJE97vVZEgWG2eSUhsyLf63Jv8+g==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/errors": "^1.3.5", + "@oclif/linewrap": "^1.0.0", + "@oclif/screen": "^1.0.4", + "ansi-escapes": "^4.3.0", + "ansi-styles": "^4.2.0", + "cardinal": "^2.1.1", + "chalk": "^4.1.0", + "clean-stack": "^3.0.0", + "cli-progress": "^3.4.0", + "extract-stack": "^2.0.0", + "fs-extra": "^8.1", + "hyperlinker": "^1.0.0", + "indent-string": "^4.0.0", + "is-wsl": "^2.2.0", + "js-yaml": "^3.13.1", + "lodash": "^4.17.21", + "natural-orderby": "^2.0.1", + "object-treeify": "^1.1.4", + "password-prompt": "^1.1.2", + "semver": "^7.3.2", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "supports-color": "^8.1.0", + "supports-hyperlinks": "^2.1.0", + "tslib": "^2.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "strip-ansi": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dev": true, + "requires": { + "ansi-regex": "^5.0.1" + } + } + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "extract-stack": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/extract-stack/-/extract-stack-2.0.0.tgz", + "integrity": "sha512-AEo4zm+TenK7zQorGK1f9mJ8L14hnTDi2ZQPR+Mub1NX8zimka1mXpV5LpH8x9HoUmFSHZCfLHqWvp0Y4FxxzQ==", + "dev": true + }, + "fs-extra": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.3.tgz", + "integrity": "sha512-q6rbdDd1o2mAnQreO7YADIxf/Whx4AHBiRf6d+/cVT8h44ss+lHgxf1FemcqDnQt9X3ct4McHr+JMGlYSsK7Cg==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "js2xmlparser": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/js2xmlparser/-/js2xmlparser-3.0.0.tgz", + "integrity": "sha1-P7YOqgicVED5MZ9RdgzNB+JJlzM=", + "dev": true, + "requires": { + "xmlcreate": "^1.0.1" + } + }, + "klaw": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/klaw/-/klaw-2.1.1.tgz", + "integrity": "sha1-QrdolHARacyRD9DRnOZ3tfs3ivE=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.9" + } + }, + "mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, + "requires": { + "ansi-regex": "^4.1.0" + }, + "dependencies": { + "ansi-regex": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", + "dev": true + } + } + }, + "supports-color": { + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", + "integrity": "sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + }, + "dependencies": { + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, + "xml2js": { + "version": "0.4.19", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.19.tgz", + "integrity": "sha512-esZnJZJOiJR9wWKMyuvSE1y6Dq5LCuJanqhxslH2bxM6duahNZ+HMpCLhBQGZkbX6xRf8x1Y2eJlgt2q3qo49Q==", + "dev": true, + "requires": { + "sax": ">=0.6.0", + "xmlbuilder": "~9.0.1" + } + }, + "xmlbuilder": { + "version": "9.0.7", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-9.0.7.tgz", + "integrity": "sha1-Ey7mPS7FVlxVfiD0wi35rKaGsQ0=", + "dev": true + }, + "xmlcreate": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/xmlcreate/-/xmlcreate-1.0.2.tgz", + "integrity": "sha1-+mv3YqYKQT+z3Y9LA8WyaSONMI8=", + "dev": true + } + } + }, "sanitize-filename-ts": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/sanitize-filename-ts/-/sanitize-filename-ts-1.0.2.tgz", @@ -9381,31 +19433,266 @@ "xmlchars": "^2.2.0" } }, + "scoped-regex": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/scoped-regex/-/scoped-regex-1.0.0.tgz", + "integrity": "sha1-o0a7Gs1CB65wvXwMfKnlZra63bg=", + "dev": true + }, "semver": { "version": "6.3.0", "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", "dev": true }, + "semver-compare": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/semver-compare/-/semver-compare-1.0.0.tgz", + "integrity": "sha1-De4hahyUGrN+nvsXiPavxf9VN/w=", + "dev": true + }, + "sentence-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/sentence-case/-/sentence-case-3.0.4.tgz", + "integrity": "sha512-8LS0JInaQMCRoQ7YUytAo/xUu5W2XnQxV2HI/6uM6U7CITS1RqPElr30V6uIqyMKM9lJGRVFy5/4CuzcixNYSg==", + "dev": true, + "requires": { + "no-case": "^3.0.4", + "tslib": "^2.0.3", + "upper-case-first": "^2.0.2" + } + }, "sequin": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/sequin/-/sequin-0.1.1.tgz", "integrity": "sha1-XC04nWajg3NOqvvEXt6ywcsb5wE=", "dev": true }, - "shebang-command": { + "set-blocking": { "version": "2.0.0", - "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", - "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", + "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", + "dev": true + }, + "set-getter": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/set-getter/-/set-getter-0.1.1.tgz", + "integrity": "sha512-9sVWOy+gthr+0G9DzqqLaYNA7+5OKkSmcqjL9cBpDEaZrr3ShQlyX2cZ/O/ozE41oxn/Tt0LGEM/w4Rub3A3gw==", + "dev": true, + "requires": { + "to-object-path": "^0.3.0" + } + }, + "set-value": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", + "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-extendable": "^0.1.1", + "is-plain-object": "^2.0.3", + "split-string": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "setimmediate": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", + "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", + "dev": true + }, + "setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==", + "dev": true + }, + "sfdx-cli": { + "version": "7.145.0", + "resolved": "https://registry.npmjs.org/sfdx-cli/-/sfdx-cli-7.145.0.tgz", + "integrity": "sha512-PCjxcBwNzsdLhrboayaK3zO2M/3JGft7sj7HoXdAFhWDDAs/LrQmKjNgss6/OrhkermPc+l9G+qgl24fDD/gcw==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.0", + "@oclif/config": "^1.17.0", + "@oclif/errors": "^1.3.4", + "@oclif/plugin-autocomplete": "^0.3.0", + "@oclif/plugin-commands": "^1.3.0", + "@oclif/plugin-help": "^3.2.2", + "@oclif/plugin-not-found": "^1.2.4", + "@oclif/plugin-plugins": "^1.10.1", + "@oclif/plugin-update": "1.5.0", + "@oclif/plugin-warn-if-update-available": "^1.7.0", + "@oclif/plugin-which": "^1.0.3", + "@salesforce/kit": "^1.5.13", + "@salesforce/lazy-require": "^0.4.0", + "@salesforce/plugin-alias": "2.0.0", + "@salesforce/plugin-apex": "0.11.0", + "@salesforce/plugin-auth": "1.8.2", + "@salesforce/plugin-community": "1.1.4", + "@salesforce/plugin-config": "1.3.24", + "@salesforce/plugin-custom-metadata": "1.0.12", + "@salesforce/plugin-data": "0.6.12", + "@salesforce/plugin-generator": "1.2.2", + "@salesforce/plugin-info": "1.3.1", + "@salesforce/plugin-limits": "1.3.0", + "@salesforce/plugin-org": "1.11.2", + "@salesforce/plugin-schema": "1.1.0", + "@salesforce/plugin-source": "1.9.2", + "@salesforce/plugin-telemetry": "1.4.0", + "@salesforce/plugin-templates": "54.3.0", + "@salesforce/plugin-trust": "^1.0.8", + "@salesforce/plugin-user": "1.7.1", + "@salesforce/require-analytics": "^0.9.16", + "@salesforce/sfdx-plugin-lwc-test": "0.1.7", + "@salesforce/ts-types": "^1.5.17", + "debug": "^4.3.1", + "salesforce-alm": "54.1.0", + "shelljs": "^0.8.4", + "tslib": "^2.1.0", + "v8-compile-cache": "^2.2.0" + }, + "dependencies": { + "@oclif/plugin-help": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/@oclif/plugin-help/-/plugin-help-3.3.1.tgz", + "integrity": "sha512-QuSiseNRJygaqAdABYFWn/H1CwIZCp9zp/PLid6yXvy6VcQV7OenEFF5XuYaCvSARe2Tg9r8Jqls5+fw1A9CbQ==", + "dev": true, + "requires": { + "@oclif/command": "^1.8.15", + "@oclif/config": "1.18.2", + "@oclif/errors": "1.3.5", + "@oclif/help": "^1.0.1", + "chalk": "^4.1.2", + "indent-string": "^4.0.0", + "lodash": "^4.17.21", + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "widest-line": "^3.1.0", + "wrap-ansi": "^6.2.0" + }, + "dependencies": { + "@oclif/config": { + "version": "1.18.2", + "resolved": "https://registry.npmjs.org/@oclif/config/-/config-1.18.2.tgz", + "integrity": "sha512-cE3qfHWv8hGRCP31j7fIS7BfCflm/BNZ2HNqHexH+fDrdF2f1D5S8VmXWLC77ffv3oDvWyvE9AZeR0RfmHCCaA==", + "dev": true, + "requires": { + "@oclif/errors": "^1.3.3", + "@oclif/parser": "^3.8.0", + "debug": "^4.1.1", + "globby": "^11.0.1", + "is-wsl": "^2.1.1", + "tslib": "^2.0.0" + } + } + } + }, + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.2.tgz", + "integrity": "sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + } + } + } + }, + "sha.js": { + "version": "2.4.11", + "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", + "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", + "dev": true, + "requires": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "shallow-clone": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", + "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", + "dev": true, + "requires": { + "kind-of": "^6.0.2" + } + }, + "shebang-command": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", + "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", "dev": true, "requires": { - "shebang-regex": "^3.0.0" + "shebang-regex": "^1.0.0" } }, "shebang-regex": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", - "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", + "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", "dev": true }, "shelljs": { @@ -9418,14 +19705,20 @@ "rechoir": "^0.6.2" } }, + "shimmer": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/shimmer/-/shimmer-1.2.1.tgz", + "integrity": "sha512-sQTKC1Re/rM6XyFM6fIAGHRPVGvyXfgzIDvzoq608vM+jeyVD0Tu1E6Np0Kc2zAIFWIj963V2800iF/9LPieQw==", + "dev": true + }, "shx": { - "version": "0.3.3", - "resolved": "https://registry.npmjs.org/shx/-/shx-0.3.3.tgz", - "integrity": "sha512-nZJ3HFWVoTSyyB+evEKjJ1STiixGztlqwKLTUNV5KqMWtGey9fTd4KU1gdZ1X9BV6215pswQ/Jew9NsuS/fNDA==", + "version": "0.3.4", + "resolved": "https://registry.npmjs.org/shx/-/shx-0.3.4.tgz", + "integrity": "sha512-N6A9MLVqjxZYcVn8hLmtneQWIJtp8IKzMP4eMnx+nqkvXoqinUPCbUFLp2UcWTEIUONhlk0ewxr/jaVGlc+J+g==", "dev": true, "requires": { "minimist": "^1.2.3", - "shelljs": "^0.8.4" + "shelljs": "^0.8.5" } }, "side-channel": { @@ -9440,9 +19733,9 @@ } }, "signal-exit": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.6.tgz", - "integrity": "sha512-sDl4qMFpijcGw22U5w63KmD3cZJfBuFlVNbVMKje2keoKML7X2UzWbc4XrmEbDwg0NXJc3yv4/ox7b+JWb57kQ==", + "version": "3.0.7", + "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz", + "integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ==", "dev": true }, "simple-concat": { @@ -9497,10 +19790,175 @@ "integrity": "sha1-VusCfWW00tzmyy4tMsTUr8nh1wc=", "dev": true }, + "smart-buffer": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.2.0.tgz", + "integrity": "sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg==", + "dev": true + }, + "snake-case": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/snake-case/-/snake-case-3.0.4.tgz", + "integrity": "sha512-LAOh4z89bGQvl9pFfNF8V146i7o7/CqFPbqzYgP+yYzDIDeS9HaNFtXABamRW+AQzEVODcvE79ljJ+8a9YSdMg==", + "dev": true, + "requires": { + "dot-case": "^3.0.4", + "tslib": "^2.0.3" + } + }, + "snapdragon": { + "version": "0.8.2", + "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", + "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", + "dev": true, + "requires": { + "base": "^0.11.1", + "debug": "^2.2.0", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "map-cache": "^0.2.2", + "source-map": "^0.5.6", + "source-map-resolve": "^0.5.0", + "use": "^3.1.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "requires": { + "ms": "2.0.0" + } + }, + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + }, + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + } + } + }, + "snapdragon-node": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", + "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", + "dev": true, + "requires": { + "define-property": "^1.0.0", + "isobject": "^3.0.0", + "snapdragon-util": "^3.0.1" + }, + "dependencies": { + "define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "requires": { + "is-descriptor": "^1.0.0" + } + }, + "is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "dev": true, + "requires": { + "kind-of": "^6.0.0" + } + }, + "is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "requires": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, + "snapdragon-util": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", + "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", + "dev": true, + "requires": { + "kind-of": "^3.2.0" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "socks": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/socks/-/socks-2.6.2.tgz", + "integrity": "sha512-zDZhHhZRY9PxRruRMR7kMhnf3I8hDs4S3f9RecfnGxvcBHQcKcIH/oUcEWffsfl1XxdYlA7nnlGbbTvPz9D8gA==", + "dev": true, + "requires": { + "ip": "^1.1.5", + "smart-buffer": "^4.2.0" + } + }, + "socks-proxy-agent": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-5.0.1.tgz", + "integrity": "sha512-vZdmnjb9a2Tz6WEQVIurybSwElwPxMZaIc7PzqbJTrezcKNznv6giT7J7tZDZ1BojVaa1jvO/UiUdhDVB0ACoQ==", + "dev": true, + "requires": { + "agent-base": "^6.0.2", + "debug": "4", + "socks": "^2.3.3" + } + }, "sort-array": { - "version": "4.1.4", - "resolved": "https://registry.npmjs.org/sort-array/-/sort-array-4.1.4.tgz", - "integrity": "sha512-GVFN6Y1sHKrWaSYOJTk9093ZnrBMc9sP3nuhANU44S4xg3rE6W5Z5WyamuT8VpMBbssnetx5faKCua0LEmUnSw==", + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/sort-array/-/sort-array-4.1.5.tgz", + "integrity": "sha512-Ya4peoS1fgFN42RN1REk2FgdNOeLIEMKFGJvs7VTP3OklF8+kl2SkpVliZ4tk/PurWsrWRsdNdU+tgyOBkB9sA==", "dev": true, "requires": { "array-back": "^5.0.0", @@ -9521,6 +19979,24 @@ } } }, + "sort-keys": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/sort-keys/-/sort-keys-2.0.0.tgz", + "integrity": "sha1-ZYU1WEhh7JfXMNbPQYIuH1ZoQSg=", + "dev": true, + "requires": { + "is-plain-obj": "^1.0.0" + } + }, + "sort-pjson": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/sort-pjson/-/sort-pjson-1.0.3.tgz", + "integrity": "sha512-h/xRX+8zuV9tbnbkRwmdNNyyJbjzVTh8YFpMgEgGU2umFDFg2EDfWKtA5YOfnBwT4YoZfJf6hrc0yuXLUvUDFA==", + "dev": true, + "requires": { + "@oclif/fixpack": "^2.3.0" + } + }, "source-map": { "version": "0.5.7", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", @@ -9533,6 +20009,19 @@ "integrity": "sha512-/3GptzWzu0+0MBQFrDKzw/DvvMTUORvgY6k6jd/VS6iCR4RDTKWH6v6WPwQoUO8667uQEf9Oe38DxAYWY5F/Ug==", "dev": true }, + "source-map-resolve": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", + "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", + "dev": true, + "requires": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } + }, "source-map-support": { "version": "0.5.21", "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.21.tgz", @@ -9551,6 +20040,18 @@ } } }, + "source-map-url": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.1.tgz", + "integrity": "sha512-cPiFOTLUKvJFIg4SKVScy4ilPPW6rFgMgfuZJPNoDuMs3nC1HbMUycBoJw77xFIp6z1UJQJOfx6C9GMH80DiTw==", + "dev": true + }, + "spawn-command": { + "version": "0.0.2-1", + "resolved": "https://registry.npmjs.org/spawn-command/-/spawn-command-0.0.2-1.tgz", + "integrity": "sha1-YvXpRmmBwbeW3Fkpk34RycaSG9A=", + "dev": true + }, "spdx-correct": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz", @@ -9583,6 +20084,15 @@ "integrity": "sha512-Ctl2BrFiM0X3MANYgj3CkygxhRmr9mi6xhejbdO960nF6EDJApTYpn0BQnDKlnNBULKiCN1n3w9EBkHK8ZWg+g==", "dev": true }, + "split-string": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", + "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", + "dev": true, + "requires": { + "extend-shallow": "^3.0.0" + } + }, "sprintf-js": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", @@ -9590,9 +20100,9 @@ "dev": true }, "sshpk": { - "version": "1.16.1", - "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", - "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", + "version": "1.17.0", + "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.17.0.tgz", + "integrity": "sha512-/9HIEs1ZXGhSPE8X6Ccm7Nam1z8KcoCqPdI7ecm1N33EzAetWahvQWVqLZtaZQ+IDKX4IyA2o0gBzqIMkAagHQ==", "dev": true, "requires": { "asn1": "~0.2.3", @@ -9606,6 +20116,12 @@ "tweetnacl": "~0.14.0" } }, + "stack-chain": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/stack-chain/-/stack-chain-1.3.7.tgz", + "integrity": "sha1-0ZLJ/06moiyUxN1FkXHj8AzqEoU=", + "dev": true + }, "stack-utils": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/stack-utils/-/stack-utils-2.0.5.tgz", @@ -9623,6 +20139,33 @@ } } }, + "static-extend": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", + "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", + "dev": true, + "requires": { + "define-property": "^0.2.5", + "object-copy": "^0.1.0" + }, + "dependencies": { + "define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "requires": { + "is-descriptor": "^0.1.0" + } + } + } + }, + "statuses": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", + "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==", + "dev": true + }, "stdout-stderr": { "version": "0.1.13", "resolved": "https://registry.npmjs.org/stdout-stderr/-/stdout-stderr-0.1.13.tgz", @@ -9659,6 +20202,12 @@ "integrity": "sha512-DBp0lSvX5G9KGRDTkR/R+a29H+Wk2xItOF+MpZLLNDWbEV9tGPnqLPxHEYjmiz8xGtJHRIqmI+hCjmNzqoA4nQ==", "dev": true }, + "strict-uri-encode": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", + "integrity": "sha1-J5siXfHVgrH1TmWt3UNS4Y+qBxM=", + "dev": true + }, "string-argv": { "version": "0.3.1", "resolved": "https://registry.npmjs.org/string-argv/-/string-argv-0.3.1.tgz", @@ -9675,6 +20224,12 @@ "strip-ansi": "^6.0.0" } }, + "string-template": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/string-template/-/string-template-0.2.1.tgz", + "integrity": "sha1-QpMuWYo1LQH8IuwzZ9nYTuxsmt0=", + "dev": true + }, "string-width": { "version": "4.2.3", "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", @@ -9713,6 +20268,14 @@ "dev": true, "requires": { "safe-buffer": "~5.1.0" + }, + "dependencies": { + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } } }, "strip-ansi": { @@ -9730,6 +20293,42 @@ "integrity": "sha512-3xurFv5tEgii33Zi8Jtp55wEIILR9eh34FAW00PZf+JnSsTmV/ioewSgQl97JHvgjoRGwPShsWm+IdrxB35d0w==", "dev": true }, + "strip-bom-buf": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-bom-buf/-/strip-bom-buf-1.0.0.tgz", + "integrity": "sha1-HLRar1dTD0yvhsf3UXnSyaUd1XI=", + "dev": true, + "requires": { + "is-utf8": "^0.2.1" + } + }, + "strip-bom-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom-stream/-/strip-bom-stream-2.0.0.tgz", + "integrity": "sha1-+H217yYT9paKpUWr/h7HKLaoKco=", + "dev": true, + "requires": { + "first-chunk-stream": "^2.0.0", + "strip-bom": "^2.0.0" + }, + "dependencies": { + "strip-bom": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-2.0.0.tgz", + "integrity": "sha1-YhmoVhZSBJHzV4i9vxRHqZx+aw4=", + "dev": true, + "requires": { + "is-utf8": "^0.2.0" + } + } + } + }, + "strip-eof": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", + "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", + "dev": true + }, "strip-final-newline": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/strip-final-newline/-/strip-final-newline-2.0.0.tgz", @@ -9742,6 +20341,12 @@ "integrity": "sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==", "dev": true }, + "strnum": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/strnum/-/strnum-1.0.5.tgz", + "integrity": "sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA==", + "dev": true + }, "supports-color": { "version": "5.5.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", @@ -9752,32 +20357,29 @@ } }, "supports-hyperlinks": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", - "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-1.0.1.tgz", + "integrity": "sha512-HHi5kVSefKaJkGYXbDuKbUGRVxqnWGn3J2e39CYcNJEfWciGq2zYtOhXLTlvrOZW1QU7VX67w7fMmWafHX9Pfw==", "dev": true, "requires": { - "has-flag": "^4.0.0", - "supports-color": "^7.0.0" + "has-flag": "^2.0.0", + "supports-color": "^5.0.0" }, "dependencies": { "has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-2.0.0.tgz", + "integrity": "sha1-6CB68cx7MNRGzHC3NLXovhj4jVE=", "dev": true - }, - "supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "requires": { - "has-flag": "^4.0.0" - } } } }, + "supports-preserve-symlinks-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", + "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", + "dev": true + }, "symbol-tree": { "version": "3.2.4", "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", @@ -9797,9 +20399,9 @@ }, "dependencies": { "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.1.tgz", + "integrity": "sha512-ILlv4k/3f6vfQ4OoP2AGvirOktlQ98ZEL1k9FaQjxa3L1abBgbuTDAdPOpvbGncC0BTVQrl+OM8xZGK6tWXt7g==", "dev": true }, "emoji-regex": { @@ -9866,6 +20468,16 @@ "integrity": "sha1-fLy2S1oUG2ou/CxdLGe04VCyomg=", "dev": true }, + "taketalk": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/taketalk/-/taketalk-1.0.0.tgz", + "integrity": "sha1-tNTw3u0gauffd1sSnqLKbeUvJt0=", + "dev": true, + "requires": { + "get-stdin": "^4.0.1", + "minimist": "^1.1.0" + } + }, "tar": { "version": "4.4.19", "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.19.tgz", @@ -9879,29 +20491,18 @@ "mkdirp": "^0.5.5", "safe-buffer": "^5.2.1", "yallist": "^3.1.1" - }, - "dependencies": { - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, - "safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "dev": true - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true - } + } + }, + "tar-fs": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.1.tgz", + "integrity": "sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==", + "dev": true, + "requires": { + "chownr": "^1.1.1", + "mkdirp-classic": "^0.5.2", + "pump": "^3.0.0", + "tar-stream": "^2.1.4" } }, "tar-stream": { @@ -9915,19 +20516,6 @@ "fs-constants": "^1.0.0", "inherits": "^2.0.3", "readable-stream": "^3.1.1" - }, - "dependencies": { - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - } } }, "temp-path": { @@ -9944,6 +20532,42 @@ "requires": { "ansi-escapes": "^4.2.1", "supports-hyperlinks": "^2.0.0" + }, + "dependencies": { + "ansi-escapes": { + "version": "4.3.2", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.2.tgz", + "integrity": "sha512-gKXj5ALrKWQLsYG9jlTRmR/xKluxHV+Z9QEwNIgCfM1/uwPMCuzVVnh5mwTd+OuBZcwSIMbqssNWRm1lE51QaQ==", + "dev": true, + "requires": { + "type-fest": "^0.21.3" + } + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + }, + "supports-hyperlinks": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/supports-hyperlinks/-/supports-hyperlinks-2.2.0.tgz", + "integrity": "sha512-6sXEzV5+I5j8Bmq9/vUphGRM/RJNT9SCURJLjwfOg51heRtguGWDzcaBlgAzKhQa0EVNpPEKzQuBwZ8S8WaCeQ==", + "dev": true, + "requires": { + "has-flag": "^4.0.0", + "supports-color": "^7.0.0" + } + } } }, "test-exclude": { @@ -9984,6 +20608,12 @@ "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", "dev": true }, + "textextensions": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/textextensions/-/textextensions-2.6.0.tgz", + "integrity": "sha512-49WtAWS+tcsy93dRt6P0P3AMD2m5PvXRhuEA0kaXos5ZLlujtYmpmFsB+QvWUSxE1ZsstmYXfQ7L40+EcQgpAQ==", + "dev": true + }, "throat": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/throat/-/throat-6.0.1.tgz", @@ -9996,6 +20626,22 @@ "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", "dev": true }, + "through2": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/through2/-/through2-3.0.2.tgz", + "integrity": "sha512-enaDQ4MUyP2W6ZyT6EsMzqBPZaM/avg8iuo+l2d3QCs0J+6RaqkHV/2/lOwDTueBHeJ/2LG9lrLW3d5rWPucuQ==", + "dev": true, + "requires": { + "inherits": "^2.0.4", + "readable-stream": "2 || 3" + } + }, + "timed-out": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/timed-out/-/timed-out-4.0.1.tgz", + "integrity": "sha1-8y6srFoXW+ol1/q1Zas+2HQe9W8=", + "dev": true + }, "tmp": { "version": "0.0.33", "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", @@ -10017,6 +20663,38 @@ "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", "dev": true }, + "to-object-path": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", + "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "to-regex": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", + "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", + "dev": true, + "requires": { + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "regex-not": "^1.0.2", + "safe-regex": "^1.1.0" + } + }, "to-regex-range": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", @@ -10026,6 +20704,12 @@ "is-number": "^7.0.0" } }, + "toidentifier": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz", + "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==", + "dev": true + }, "tough-cookie": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-4.0.0.tgz", @@ -10046,6 +20730,18 @@ "punycode": "^2.1.1" } }, + "traverse": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/traverse/-/traverse-0.3.9.tgz", + "integrity": "sha1-cXuPIgzAu3tE5AUUwisui7xw2Lk=", + "dev": true + }, + "tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "dev": true + }, "treeify": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/treeify/-/treeify-1.1.0.tgz", @@ -10075,20 +20771,20 @@ } }, "ts-retry-promise": { - "version": "0.6.0", - "resolved": "https://registry.npmjs.org/ts-retry-promise/-/ts-retry-promise-0.6.0.tgz", - "integrity": "sha512-8DF80uA7JPu6m8ouHxGkyBpPTIGQnsgIUgLDqcRaD7EEhVowjG72KqCX334gsa1P+AmzeTVdd/xEzVFCAuPCtg==", + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/ts-retry-promise/-/ts-retry-promise-0.6.1.tgz", + "integrity": "sha512-6L9PAWahkRtZ4mG48wz3Mxk7LfW1eZKEPsCteIa5fbDE1G2kFk4ThHXbynKlIZLg0RdenDBDw6CLME5liOrBSQ==", "dev": true }, "tsconfig-paths": { - "version": "3.12.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.12.0.tgz", - "integrity": "sha512-e5adrnOYT6zqVnWqZu7i/BQ3BnhzvGbjEjejFXO20lKIKpwTaupkCPgEfv4GZK1IBciJUEhYs3J3p75FdaTFVg==", + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.14.1.tgz", + "integrity": "sha512-fxDhWnFSLt3VuTwtvJt5fpwxBHg5AdKWMsgcPOOIilyjymcYVZoCQF8fvFRezCNfblEXmi+PcM1eYHeOAgXCOQ==", "dev": true, "requires": { "@types/json5": "^0.0.29", "json5": "^1.0.1", - "minimist": "^1.2.0", + "minimist": "^1.2.6", "strip-bom": "^3.0.0" }, "dependencies": { @@ -10212,9 +20908,9 @@ "integrity": "sha512-8Y75pvTYkLJW2hWQHXxoqRgV7qb9B+9vFEtidML+7koHUFapnVJAZ6cKs+Qjz5Aw3aZWHMC6u0wJE3At+nSGwA==" }, "uglify-js": { - "version": "3.14.5", - "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.14.5.tgz", - "integrity": "sha512-qZukoSxOG0urUTvjc2ERMTcAy+BiFh3weWAkeurLwjrCba73poHmG3E36XEjd/JGukMzwTL7uCxZiAexj8ppvQ==", + "version": "3.15.4", + "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.15.4.tgz", + "integrity": "sha512-vMOPGDuvXecPs34V74qDKk4iJ/SN4vL3Ow/23ixafENYvtrNvtbcgUeugTcUGRGsOF/5fU8/NYSL5Hyb3l1OJA==", "dev": true, "optional": true }, @@ -10236,18 +20932,153 @@ "integrity": "sha512-ekY1NhRzq0B08g4bGuX4wd2jZx5GnKz6mKSqFL4nqBlfyMGiG10gDFhDTMEfYmDL6Jy0FUIZp7wiRB+0BP7J2g==", "dev": true }, + "union-value": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", + "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", + "dev": true, + "requires": { + "arr-union": "^3.1.0", + "get-value": "^2.0.6", + "is-extendable": "^0.1.1", + "set-value": "^2.0.1" + } + }, "universalify": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", "dev": true }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", + "dev": true + }, + "unset-value": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", + "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", + "dev": true, + "requires": { + "has-value": "^0.3.1", + "isobject": "^3.0.0" + }, + "dependencies": { + "has-value": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", + "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", + "dev": true, + "requires": { + "get-value": "^2.0.3", + "has-values": "^0.1.4", + "isobject": "^2.0.0" + }, + "dependencies": { + "isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "dev": true, + "requires": { + "isarray": "1.0.0" + } + } + } + }, + "has-values": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", + "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", + "dev": true + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + } + } + }, "untildify": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/untildify/-/untildify-4.0.0.tgz", "integrity": "sha512-KK8xQ1mkzZeg9inewmFVDNkg3l5LUhoq9kN6iWYB/CC9YMG8HA+c1Q8HwDe6dEX7kErrEVNVBO3fWsVq5iDgtw==", "dev": true }, + "unzip-response": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/unzip-response/-/unzip-response-2.0.1.tgz", + "integrity": "sha1-0vD3N9FrBhXnKmk17QQhRXLVb5c=", + "dev": true + }, + "unzipper": { + "version": "0.10.11", + "resolved": "https://registry.npmjs.org/unzipper/-/unzipper-0.10.11.tgz", + "integrity": "sha512-+BrAq2oFqWod5IESRjL3S8baohbevGcVA+teAIOYWM3pDVdseogqbzhhvvmiyQrUNKFUnDMtELW3X8ykbyDCJw==", + "dev": true, + "requires": { + "big-integer": "^1.6.17", + "binary": "~0.3.0", + "bluebird": "~3.4.1", + "buffer-indexof-polyfill": "~1.0.0", + "duplexer2": "~0.1.4", + "fstream": "^1.0.12", + "graceful-fs": "^4.2.2", + "listenercount": "~1.0.1", + "readable-stream": "~2.3.6", + "setimmediate": "~1.0.4" + }, + "dependencies": { + "bluebird": { + "version": "3.4.7", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.4.7.tgz", + "integrity": "sha1-9y12C+Cbf3bQjtj66Ysomo0F+rM=", + "dev": true + }, + "readable-stream": { + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", + "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", + "dev": true, + "requires": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", + "dev": true + } + } + }, + "upper-case": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/upper-case/-/upper-case-2.0.2.tgz", + "integrity": "sha512-KgdgDGJt2TpuwBUIjgG6lzw2GWFRCW9Qkfkiv0DxqHHLYJHmtmdUIKcZd8rHgFSjopVTlw6ggzCm1b8MFQwikg==", + "dev": true, + "requires": { + "tslib": "^2.0.3" + } + }, + "upper-case-first": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/upper-case-first/-/upper-case-first-2.0.2.tgz", + "integrity": "sha512-514ppYHBaKwfJRK/pNC6c/OxfGa0obSnAl106u97Ed0I625Nin96KAjttZF6ZL3e1XLtphxnqrOi9iWgm+u+bg==", + "dev": true, + "requires": { + "tslib": "^2.0.3" + } + }, "uri-js": { "version": "4.4.1", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.1.tgz", @@ -10257,6 +21088,33 @@ "punycode": "^2.1.0" } }, + "urix": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", + "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", + "dev": true + }, + "url-parse-lax": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/url-parse-lax/-/url-parse-lax-1.0.0.tgz", + "integrity": "sha1-evjzA2Rem9eaJy56FKxovAYJ2nM=", + "dev": true, + "requires": { + "prepend-http": "^1.0.1" + } + }, + "url-to-options": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/url-to-options/-/url-to-options-1.0.1.tgz", + "integrity": "sha1-FQWgOiiaSMvXpDTvuu7FBV9WM6k=", + "dev": true + }, + "use": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", + "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", + "dev": true + }, "utf8-byte-length": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/utf8-byte-length/-/utf8-byte-length-1.0.4.tgz", @@ -10288,9 +21146,9 @@ "dev": true }, "v8-to-istanbul": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-8.1.0.tgz", - "integrity": "sha512-/PRhfd8aTNp9Ggr62HPzXg2XasNFGy5PBt0Rp04du7/8GNNSgxFL6WBTkgMKSL9bFjH+8kKEG3f37FmxiTqUUA==", + "version": "8.1.1", + "resolved": "https://registry.npmjs.org/v8-to-istanbul/-/v8-to-istanbul-8.1.1.tgz", + "integrity": "sha512-FGtKtv3xIpR6BYhvgH8MI/y78oT7d8Au3ww4QIxymrCtZEh5b8gCw2siywE+puhEmuWKDtmfrvF5UlB298ut3w==", "dev": true, "requires": { "@types/istanbul-lib-coverage": "^2.0.1", @@ -10335,6 +21193,57 @@ } } }, + "vinyl": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/vinyl/-/vinyl-2.2.1.tgz", + "integrity": "sha512-LII3bXRFBZLlezoG5FfZVcXflZgWP/4dCwKtxd5ky9+LOtM4CS3bIRQsmR1KMnMW07jpE8fqR2lcxPZ+8sJIcw==", + "dev": true, + "requires": { + "clone": "^2.1.1", + "clone-buffer": "^1.0.0", + "clone-stats": "^1.0.0", + "cloneable-readable": "^1.0.0", + "remove-trailing-separator": "^1.0.1", + "replace-ext": "^1.0.0" + } + }, + "vinyl-file": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/vinyl-file/-/vinyl-file-3.0.0.tgz", + "integrity": "sha1-sQTZ5ECf+jJfqt1SBkLQo7SIs2U=", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "pify": "^2.3.0", + "strip-bom-buf": "^1.0.0", + "strip-bom-stream": "^2.0.0", + "vinyl": "^2.0.1" + } + }, + "vm2": { + "version": "3.9.9", + "resolved": "https://registry.npmjs.org/vm2/-/vm2-3.9.9.tgz", + "integrity": "sha512-xwTm7NLh/uOjARRBs8/95H0e8fT3Ukw5D/JJWhxMbhKzNh1Nu981jQKvkep9iKYNxzlVrdzD0mlBGkDKZWprlw==", + "dev": true, + "requires": { + "acorn": "^8.7.0", + "acorn-walk": "^8.2.0" + }, + "dependencies": { + "acorn": { + "version": "8.7.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.7.0.tgz", + "integrity": "sha512-V/LGr1APy+PXIwKebEWrkZPwoeoF+w1jiOBUmuxuiUIaOHtob8Qc9BTrYo7VuI5fR8tqsy+buA2WFooR5olqvQ==", + "dev": true + }, + "acorn-walk": { + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz", + "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==", + "dev": true + } + } + }, "w3c-hr-time": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/w3c-hr-time/-/w3c-hr-time-1.0.2.tgz", @@ -10437,9 +21346,9 @@ } }, "which": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", - "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", "dev": true, "requires": { "isexe": "^2.0.0" @@ -10458,6 +21367,12 @@ "is-symbol": "^1.0.3" } }, + "which-module": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", + "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", + "dev": true + }, "widest-line": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/widest-line/-/widest-line-3.1.0.tgz", @@ -10473,6 +21388,25 @@ "integrity": "sha1-ugZWKbepJRMOFXeRCM9UCZDpjRs=", "dev": true }, + "with-open-file": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/with-open-file/-/with-open-file-0.1.7.tgz", + "integrity": "sha512-ecJS2/oHtESJ1t3ZfMI3B7KIDKyfN0O16miWxdn30zdh66Yd3LsRFebXZXq6GU4xfxLf6nVxp9kIqElb5fqczA==", + "dev": true, + "requires": { + "p-finally": "^1.0.0", + "p-try": "^2.1.0", + "pify": "^4.0.1" + }, + "dependencies": { + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + } + } + }, "word-wrap": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", @@ -10495,168 +21429,923 @@ "typical": "^2.6.1" } }, - "wrap-ansi": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", - "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "requires": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + } + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" + }, + "write": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", + "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", + "dev": true, + "requires": { + "mkdirp": "^0.5.1" + } + }, + "write-file-atomic": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", + "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", + "dev": true, + "requires": { + "imurmurhash": "^0.1.4", + "is-typedarray": "^1.0.0", + "signal-exit": "^3.0.2", + "typedarray-to-buffer": "^3.1.5" + } + }, + "ws": { + "version": "7.5.7", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.7.tgz", + "integrity": "sha512-KMvVuFzpKBuiIXW3E4u3mySRO2/mCHSyZDJQM5NQ9Q9KHWHWh0NHgfbRMLLrceUK5qAL4ytALJbpRMjixFZh8A==", + "dev": true + }, + "xml-js": { + "version": "1.6.11", + "resolved": "https://registry.npmjs.org/xml-js/-/xml-js-1.6.11.tgz", + "integrity": "sha512-7rVi2KMfwfWFl+GpPg6m80IVMWXLRjO+PxTq7V2CDhoGak0wzYzFgUY2m4XJ47OGdXd8eLE8EmwfAmdjw7lC1g==", + "dev": true, + "requires": { + "sax": "^1.2.4" + } + }, + "xml-name-validator": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", + "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==", + "dev": true + }, + "xml2js": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", + "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", + "dev": true, + "requires": { + "sax": ">=0.6.0", + "xmlbuilder": "~11.0.0" + } + }, + "xmlbuilder": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", + "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", + "dev": true + }, + "xmlchars": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", + "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", + "dev": true + }, + "xmlcreate": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/xmlcreate/-/xmlcreate-2.0.4.tgz", + "integrity": "sha512-nquOebG4sngPmGPICTS5EnxqhKbCmz5Ox5hsszI2T6U5qdrJizBc+0ilYSEjTSzU0yZcmvppztXe/5Al5fUwdg==", + "dev": true + }, + "xmldom-sfdx-encoding": { + "version": "0.1.30", + "resolved": "https://registry.npmjs.org/xmldom-sfdx-encoding/-/xmldom-sfdx-encoding-0.1.30.tgz", + "integrity": "sha512-NOZCfMfwvCMBlSMBr971cnjmToNswV68A1CA3pnM0WGauo1BhWpTgSsj6Lbq8HNAI2OOdWktCSMLtaZU5wVBHA==", + "dev": true + }, + "xregexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/xregexp/-/xregexp-2.0.0.tgz", + "integrity": "sha1-UqY+VsoLhKfzpfPWGHLxJq16WUM=", + "dev": true + }, + "y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true + }, + "yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + }, + "yaml": { + "version": "1.10.2", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", + "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", + "dev": true + }, + "yargs": { + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", + "dev": true, + "requires": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + } + }, + "yargs-parser": { + "version": "20.2.9", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", + "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", + "dev": true + }, + "yarn": { + "version": "1.22.18", + "resolved": "https://registry.npmjs.org/yarn/-/yarn-1.22.18.tgz", + "integrity": "sha512-oFffv6Jp2+BTUBItzx1Z0dpikTX+raRdqupfqzeMKnoh7WD6RuPAxcqDkMUy9vafJkrB0YaV708znpuMhEBKGQ==", + "dev": true + }, + "yeoman-environment": { + "version": "2.10.3", + "resolved": "https://registry.npmjs.org/yeoman-environment/-/yeoman-environment-2.10.3.tgz", + "integrity": "sha512-pLIhhU9z/G+kjOXmJ2bPFm3nejfbH+f1fjYRSOteEXDBrv1EoJE/e+kuHixSXfCYfTkxjYsvRaDX+1QykLCnpQ==", + "dev": true, + "requires": { + "chalk": "^2.4.1", + "debug": "^3.1.0", + "diff": "^3.5.0", + "escape-string-regexp": "^1.0.2", + "execa": "^4.0.0", + "globby": "^8.0.1", + "grouped-queue": "^1.1.0", + "inquirer": "^7.1.0", + "is-scoped": "^1.0.0", + "lodash": "^4.17.10", + "log-symbols": "^2.2.0", + "mem-fs": "^1.1.0", + "mem-fs-editor": "^6.0.0", + "npm-api": "^1.0.0", + "semver": "^7.1.3", + "strip-ansi": "^4.0.0", + "text-table": "^0.2.0", + "untildify": "^3.0.3", + "yeoman-generator": "^4.8.2" + }, + "dependencies": { + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true + }, + "ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "cross-spawn": { + "version": "7.0.3", + "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.3.tgz", + "integrity": "sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==", + "dev": true, + "requires": { + "path-key": "^3.1.0", + "shebang-command": "^2.0.0", + "which": "^2.0.1" + } + }, + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "requires": { + "ms": "^2.1.1" + } + }, + "diff": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/diff/-/diff-3.5.0.tgz", + "integrity": "sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==", + "dev": true + }, + "dir-glob": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.0.0.tgz", + "integrity": "sha512-37qirFDz8cA5fimp9feo43fSuRo2gHwaIn6dXL8Ber1dGwUosDrGZeCCXq57WnIqE4aQ+u3eQZzsk1yOzhdwag==", + "dev": true, + "requires": { + "arrify": "^1.0.1", + "path-type": "^3.0.0" + } + }, + "execa": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-4.1.0.tgz", + "integrity": "sha512-j5W0//W7f8UxAn8hXVnwG8tLwdiUy4FJLcSupCg6maBYZDpyBvTApK7KyuI4bKj8KOh1r2YH+6ucuYtJv1bTZA==", + "dev": true, + "requires": { + "cross-spawn": "^7.0.0", + "get-stream": "^5.0.0", + "human-signals": "^1.1.1", + "is-stream": "^2.0.0", + "merge-stream": "^2.0.0", + "npm-run-path": "^4.0.0", + "onetime": "^5.1.0", + "signal-exit": "^3.0.2", + "strip-final-newline": "^2.0.0" + } + }, + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "get-stream": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-5.2.0.tgz", + "integrity": "sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA==", + "dev": true, + "requires": { + "pump": "^3.0.0" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "8.0.2", + "resolved": "https://registry.npmjs.org/globby/-/globby-8.0.2.tgz", + "integrity": "sha512-yTzMmKygLp8RUpG1Ymu2VXPSJQZjNAZPD4ywgYEaG7e4tBJeUQBO8OpXrf1RCNcEs5alsoJYPAMiIHP0cmeC7w==", + "dev": true, + "requires": { + "array-union": "^1.0.1", + "dir-glob": "2.0.0", + "fast-glob": "^2.0.2", + "glob": "^7.1.2", + "ignore": "^3.3.5", + "pify": "^3.0.0", + "slash": "^1.0.0" + } + }, + "human-signals": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/human-signals/-/human-signals-1.1.1.tgz", + "integrity": "sha512-SEQu7vl8KjNL2eoGBLF3+wAjpsNfA9XMlXAYj/3EdaNfAlxKthD1xjEQfGOUhllCGGJVNY34bRr6lPINhNjyZw==", + "dev": true + }, + "ignore": { + "version": "3.3.10", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-3.3.10.tgz", + "integrity": "sha512-Pgs951kaMm5GXP7MOvxERINe3gsaVjUWFm+UZPSq9xYriQAksyhg0csnS0KXSNRD5NmNdapXEpjxG49+AKh/ug==", + "dev": true + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + } + }, + "path-key": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/path-key/-/path-key-3.1.1.tgz", + "integrity": "sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q==", + "dev": true + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + } + }, + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "shebang-command": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", + "integrity": "sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA==", + "dev": true, + "requires": { + "shebang-regex": "^3.0.0" + } + }, + "shebang-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-3.0.0.tgz", + "integrity": "sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A==", + "dev": true + }, + "slash": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-1.0.0.tgz", + "integrity": "sha1-xB8vbDn8FtHNF61LXYlhFK5HDVU=", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", + "dev": true, + "requires": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + } + }, + "untildify": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/untildify/-/untildify-3.0.3.tgz", + "integrity": "sha512-iSk/J8efr8uPT/Z4eSUywnqyrQU7DSdMfdqK4iWEaUVVmcP5JcnpRqmVMwcwcnmI1ATFNgC5V90u09tBynNFKA==", + "dev": true + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, + "yeoman-generator": { + "version": "4.13.0", + "resolved": "https://registry.npmjs.org/yeoman-generator/-/yeoman-generator-4.13.0.tgz", + "integrity": "sha512-f2/5N5IR3M2Ozm+QocvZQudlQITv2DwI6Mcxfy7R7gTTzaKgvUpgo/pQMJ+WQKm0KN0YMWCFOZpj0xFGxevc1w==", "dev": true, "requires": { - "ansi-styles": "^4.0.0", - "string-width": "^4.1.0", - "strip-ansi": "^6.0.0" + "async": "^2.6.2", + "chalk": "^2.4.2", + "cli-table": "^0.3.1", + "cross-spawn": "^6.0.5", + "dargs": "^6.1.0", + "dateformat": "^3.0.3", + "debug": "^4.1.1", + "diff": "^4.0.1", + "error": "^7.0.2", + "find-up": "^3.0.0", + "github-username": "^3.0.0", + "grouped-queue": "^1.1.0", + "istextorbinary": "^2.5.1", + "lodash": "^4.17.11", + "make-dir": "^3.0.0", + "mem-fs-editor": "^7.0.1", + "minimist": "^1.2.5", + "pretty-bytes": "^5.2.0", + "read-chunk": "^3.2.0", + "read-pkg-up": "^5.0.0", + "rimraf": "^2.6.3", + "run-async": "^2.0.0", + "semver": "^7.2.1", + "shelljs": "^0.8.4", + "text-table": "^0.2.0", + "through2": "^3.0.1", + "yeoman-environment": "^2.9.5" }, "dependencies": { - "ansi-styles": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", - "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "@nodelib/fs.stat": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", + "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", + "dev": true + }, + "array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "requires": { + "array-uniq": "^1.0.1" + } + }, + "async": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", + "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", + "dev": true, + "requires": { + "lodash": "^4.17.14" + } + }, + "braces": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", + "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", + "dev": true, + "requires": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "dir-glob": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-2.2.2.tgz", + "integrity": "sha512-f9LBi5QWzIW3I6e//uxZoLBlUt9kcp66qo0sSCxL6YZKc75R1c4MFCoe/LaZiBGmgujvQdxc5Bn3QhfyvK5Hsw==", + "dev": true, + "requires": { + "path-type": "^3.0.0" + } + }, + "fast-glob": { + "version": "2.2.7", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", + "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", + "dev": true, + "requires": { + "@mrmlnc/readdir-enhanced": "^2.2.1", + "@nodelib/fs.stat": "^1.1.2", + "glob-parent": "^3.1.0", + "is-glob": "^4.0.0", + "merge2": "^1.2.3", + "micromatch": "^3.1.10" + } + }, + "fill-range": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", + "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", + "dev": true, + "requires": { + "extend-shallow": "^2.0.1", + "is-number": "^3.0.0", + "repeat-string": "^1.6.1", + "to-regex-range": "^2.1.0" + }, + "dependencies": { + "extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "requires": { + "is-extendable": "^0.1.0" + } + } + } + }, + "find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "requires": { + "locate-path": "^3.0.0" + } + }, + "glob-parent": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", + "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", + "dev": true, + "requires": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + }, + "dependencies": { + "is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, + "requires": { + "is-extglob": "^2.1.0" + } + } + } + }, + "globby": { + "version": "9.2.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-9.2.0.tgz", + "integrity": "sha512-ollPHROa5mcxDEkwg6bPt3QbEf4pDQSNtd6JPL1YvOvAo/7/0VAm9TccUeoTmarjPw4pfUthSCqcyfNB1I3ZSg==", + "dev": true, + "requires": { + "@types/glob": "^7.1.1", + "array-union": "^1.0.2", + "dir-glob": "^2.2.2", + "fast-glob": "^2.2.6", + "glob": "^7.1.3", + "ignore": "^4.0.3", + "pify": "^4.0.1", + "slash": "^2.0.0" + } + }, + "ignore": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-4.0.6.tgz", + "integrity": "sha512-cyFDKrqc/YdcWFniJhzI42+AzS+gNwmUzOSFcRCQYwySuBBBy/KjuxWLZ/FHEH6Moq1NizMOBWyTcv8O4OZIMg==", + "dev": true + }, + "is-number": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", + "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "dependencies": { + "kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "requires": { + "is-buffer": "^1.1.5" + } + } + } + }, + "isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true + }, + "locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "requires": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + } + }, + "mem-fs-editor": { + "version": "7.1.0", + "resolved": "https://registry.npmjs.org/mem-fs-editor/-/mem-fs-editor-7.1.0.tgz", + "integrity": "sha512-BH6QEqCXSqGeX48V7zu+e3cMwHU7x640NB8Zk8VNvVZniz+p4FK60pMx/3yfkzo6miI6G3a8pH6z7FeuIzqrzA==", + "dev": true, + "requires": { + "commondir": "^1.0.1", + "deep-extend": "^0.6.0", + "ejs": "^3.1.5", + "glob": "^7.1.4", + "globby": "^9.2.0", + "isbinaryfile": "^4.0.0", + "mkdirp": "^1.0.0", + "multimatch": "^4.0.0", + "rimraf": "^3.0.0", + "through2": "^3.0.2", + "vinyl": "^2.2.1" + }, + "dependencies": { + "rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + } + } + }, + "micromatch": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", + "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", + "dev": true, + "requires": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + } + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", "dev": true, "requires": { - "color-convert": "^2.0.1" + "p-try": "^2.0.0" } }, - "color-convert": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", - "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", "dev": true, "requires": { - "color-name": "~1.1.4" + "p-limit": "^2.0.0" } }, - "color-name": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "path-exists": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", + "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", "dev": true - } - } - }, - "wrappy": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" - }, - "write": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/write/-/write-1.0.3.tgz", - "integrity": "sha512-/lg70HAjtkUgWPVZhZcm+T4hkL8Zbtp1nFNOn3lRrxnlv50SRBv7cR7RqR+GMsd3hUXy9hWBo4CHTbFTcOYwig==", - "dev": true, - "requires": { - "mkdirp": "^0.5.1" - }, - "dependencies": { - "mkdirp": { - "version": "0.5.5", - "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", - "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", + }, + "path-type": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", + "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", + "dev": true, + "requires": { + "pify": "^3.0.0" + }, + "dependencies": { + "pify": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", + "integrity": "sha1-5aSs0sEB/fPZpNB/DbxNtJ3SgXY=", + "dev": true + } + } + }, + "pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true + }, + "read-pkg": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-5.2.0.tgz", + "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", + "dev": true, + "requires": { + "@types/normalize-package-data": "^2.4.0", + "normalize-package-data": "^2.5.0", + "parse-json": "^5.0.0", + "type-fest": "^0.6.0" + } + }, + "read-pkg-up": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-5.0.0.tgz", + "integrity": "sha512-XBQjqOBtTzyol2CpsQOw8LHV0XbDZVG7xMMjmXAJomlVY03WOBRmYgDJETlvcg0H63AJvPRwT7GFi5rvOzUOKg==", + "dev": true, + "requires": { + "find-up": "^3.0.0", + "read-pkg": "^5.0.0" + } + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "semver": { + "version": "7.3.7", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.7.tgz", + "integrity": "sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "slash": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-2.0.0.tgz", + "integrity": "sha512-ZYKh3Wh2z1PpEXWr0MpSBZ0V6mZHAQfYevttO11c51CaWjGTaadiKZ+wVt1PbMlDV5qhMFslpZCemhwOK7C89A==", + "dev": true + }, + "to-regex-range": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", + "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", "dev": true, "requires": { - "minimist": "^1.2.5" + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" } + }, + "type-fest": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.6.0.tgz", + "integrity": "sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==", + "dev": true } } }, - "write-file-atomic": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/write-file-atomic/-/write-file-atomic-3.0.3.tgz", - "integrity": "sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q==", - "dev": true, - "requires": { - "imurmurhash": "^0.1.4", - "is-typedarray": "^1.0.0", - "signal-exit": "^3.0.2", - "typedarray-to-buffer": "^3.1.5" - } - }, - "ws": { - "version": "7.5.6", - "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.6.tgz", - "integrity": "sha512-6GLgCqo2cy2A2rjCNFlxQS6ZljG/coZfZXclldI8FB/1G3CCI36Zd8xy2HrFVACi8tfk5XrgLQEk+P0Tnz9UcA==", - "dev": true - }, - "xml-js": { - "version": "1.6.11", - "resolved": "https://registry.npmjs.org/xml-js/-/xml-js-1.6.11.tgz", - "integrity": "sha512-7rVi2KMfwfWFl+GpPg6m80IVMWXLRjO+PxTq7V2CDhoGak0wzYzFgUY2m4XJ47OGdXd8eLE8EmwfAmdjw7lC1g==", - "dev": true, - "requires": { - "sax": "^1.2.4" - } - }, - "xml-name-validator": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", - "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==", - "dev": true - }, - "xml2js": { - "version": "0.4.23", - "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", - "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", - "dev": true, - "requires": { - "sax": ">=0.6.0", - "xmlbuilder": "~11.0.0" - } - }, - "xmlbuilder": { - "version": "11.0.1", - "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", - "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", - "dev": true - }, - "xmlchars": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", - "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", - "dev": true - }, - "xmlcreate": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/xmlcreate/-/xmlcreate-2.0.4.tgz", - "integrity": "sha512-nquOebG4sngPmGPICTS5EnxqhKbCmz5Ox5hsszI2T6U5qdrJizBc+0ilYSEjTSzU0yZcmvppztXe/5Al5fUwdg==", - "dev": true - }, - "y18n": { - "version": "5.0.8", - "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", - "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", - "dev": true - }, - "yallist": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", - "dev": true - }, - "yaml": { - "version": "1.10.2", - "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", - "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", - "dev": true - }, - "yargs": { - "version": "16.2.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", - "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", - "dev": true, - "requires": { - "cliui": "^7.0.2", - "escalade": "^3.1.1", - "get-caller-file": "^2.0.5", - "require-directory": "^2.1.1", - "string-width": "^4.2.0", - "y18n": "^5.0.5", - "yargs-parser": "^20.2.2" - } - }, - "yargs-parser": { - "version": "20.2.9", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", - "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", - "dev": true - }, "yn": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", @@ -10669,6 +22358,132 @@ "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", "dev": true }, + "yosay": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/yosay/-/yosay-2.0.2.tgz", + "integrity": "sha512-avX6nz2esp7IMXGag4gu6OyQBsMh/SEn+ZybGu3yKPlOTE6z9qJrzG/0X5vCq/e0rPFy0CUYCze0G5hL310ibA==", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0", + "ansi-styles": "^3.0.0", + "chalk": "^1.0.0", + "cli-boxes": "^1.0.0", + "pad-component": "0.0.1", + "string-width": "^2.0.0", + "strip-ansi": "^3.0.0", + "taketalk": "^1.0.0", + "wrap-ansi": "^2.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true + }, + "chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "requires": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" + }, + "dependencies": { + "ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true + } + } + }, + "is-fullwidth-code-point": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", + "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", + "dev": true + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.1.tgz", + "integrity": "sha512-+O9Jct8wf++lXxxFc4hc8LsjaSq0HFzzL7cVsw8pRDIPdjKD2mT4ytDZlLuSBZ4cLKZFXIrMGO7DbQCtMJJMKw==", + "dev": true + }, + "strip-ansi": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", + "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", + "dev": true, + "requires": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "requires": { + "ansi-regex": "^2.0.0" + } + }, + "supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true + }, + "wrap-ansi": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-2.1.0.tgz", + "integrity": "sha1-2Pw9KE3QV5T+hJc8rs3Rz4JP3YU=", + "dev": true, + "requires": { + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1" + }, + "dependencies": { + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + } + } + } + } + }, "zip-stream": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/zip-stream/-/zip-stream-4.1.0.tgz", @@ -10678,19 +22493,6 @@ "archiver-utils": "^2.1.0", "compress-commons": "^4.1.0", "readable-stream": "^3.6.0" - }, - "dependencies": { - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - } } } } diff --git a/package.json b/package.json index a25333e03..662b8009d 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "nebula-logger", - "version": "4.7.0", + "version": "4.7.1", "description": "Designed for Salesforce admins, developers & architects. A robust logger for Apex, Flow, Process Builder & Integrations.", "author": "Jonathan Gillespie", "license": "MIT", @@ -22,7 +22,8 @@ "flow", "lightning components", "lwc", - "aura" + "aura", + "big objects" ], "engines": { "node": ">= 12.18.0", @@ -37,7 +38,7 @@ "docs:fix:lwc": "pwsh ./scripts/build/generate-lwc-docs.ps1 && git add ./docs/lightning-components", "docs:verify": "pwsh ./scripts/build/verify-docs-up-to-date.ps1", "experience:create": "sfdx force:community:create --name \"Logger Test Site\" --templatename \"Customer Service\" --urlpathprefix logger --description \"Logger Test Site\"", - "experience:deploy": "sfdx force:source:deploy --sourcepath ./nebula-logger/experience-cloud --wait 30", + "experience:deploy": "sfdx force:source:deploy --sourcepath ./config/experience-cloud --wait 30", "husky:pre-commit": "lint-staged --config ./config/linters/lint-staged.config.js", "lint:fix:lwc": "eslint --config ./config/linters/.eslintrc.json **/lwc/** --fix && eslint --config ./config/linters/.eslintrc.json **/aura/** --fix", "lint:verify": "npm run lint:verify:apex && npm run lint:verify:lwc", @@ -53,7 +54,11 @@ "package:version:create:unlocked": "sfdx force:package:version:create --json --package \"Nebula Logger - Core\" --skipancestorcheck --codecoverage --installationkeybypass --wait 30", "package:version:number:fix": "pwsh ./scripts/build/sync-package-version-number.ps1", "package:version:number:verify": "pwsh ./scripts/build/validate-current-package-version-number.ps1", + "permset:assign": "npm run permset:assign:admin && npm run permset:assign:big-object-admin && npm run permset:assign:slack-admin", "permset:assign:admin": "sfdx force:user:permset:assign --permsetname LoggerAdmin", + "permset:assign:big-object-admin": "sfdx force:user:permset:assign --permsetname LoggerLogEntryArchiveAdmin", + "permset:assign:slack-admin": "sfdx force:user:permset:assign --permsetname LoggerSlackPluginAdmin", + "plugin:version:create:big-object-archiving": "sfdx force:package:version:create --json --package \"Nebula Logger - Core Plugin - Big Object Archiving\" --codecoverage --installationkeybypass --wait 30", "plugin:version:create:log-retention-rules": "sfdx force:package:version:create --json --package \"Nebula Logger - Core Plugin - Log Retention Rules\" --codecoverage --installationkeybypass --wait 30", "plugin:version:create:logger-admin-dashboard": "sfdx force:package:version:create --json --package \"Nebula Logger - Core Plugin - Logger Admin Dashboard\" --codecoverage --installationkeybypass --wait 30", "plugin:version:create:slack": "sfdx force:package:version:create --json --package \"Nebula Logger - Core Plugin - Slack\" --codecoverage --installationkeybypass --wait 30", @@ -63,11 +68,14 @@ "sfdx:plugins:link:bummer": "npx sfdx plugins:link ./node_modules/@jongpie/sfdx-bummer-plugin", "sfdx:plugins:link:prettier": "npx sfdx plugins:link ./node_modules/@jayree/sfdx-plugin-prettier", "sfdx:plugins:link:scanner": "npx sfdx plugins:link ./node_modules/@salesforce/sfdx-scanner", + "source:deploy": "sfdx force:source:deploy --sourcepath ./nebula-logger/ --wait 30", + "source:deploy:core": "sfdx force:source:deploy --sourcepath ./nebula-logger/core --wait 30", "source:push": "sfdx force:source:push --forceoverwrite", "source:push:test": "npm run source:push && npm run test:apex", "test": "npm run test:lwc && npm run test:apex", "test:apex": "sfdx force:apex:test:run --verbose --testlevel RunLocalTests --wait 30 --resultformat human --codecoverage --detailedcoverage --outputdir ./test-coverage/apex", - "test:apex:suites": "sfdx force:apex:test:run --verbose --suitenames LoggerConfiguration,LoggerEngine,LoggerLogManagement,LoggerPluginFramework --wait 30 --resultformat human --codecoverage --detailedcoverage --outputdir ./tests/apex", + "test:apex:without-coverage": "sfdx force:apex:test:run --verbose --testlevel RunLocalTests --wait 30 --resultformat human --outputdir ./test-coverage/apex", + "test:apex:suites": "sfdx force:apex:test:run --verbose --suitenames LoggerConfiguration,LoggerEngine,LoggerLogManagement --wait 30 --resultformat human --codecoverage --detailedcoverage --outputdir ./tests/apex", "test:lwc": "sfdx-lwc-jest --coverage --skipApiVersionCheck --verbose" }, "dependencies": { @@ -81,7 +89,7 @@ "devDependencies": { "@babel/core": "latest", "@babel/eslint-parser": "latest", - "@cparra/apexdocs": "latest", + "@cparra/apexdocs": "1.13.7", "@jayree/sfdx-plugin-prettier": "latest", "@jongpie/sfdx-bummer-plugin": "latest", "@ljharb/eslint-config": "latest", @@ -100,7 +108,7 @@ "lint-staged": "latest", "prettier": "latest", "prettier-plugin-apex": "latest", - "pwsh": "latest", + "pwsh": "^0.3.0", "sfdx-cli": "latest" } } diff --git a/scripts/build/create-and-install-package-version.ps1 b/scripts/build/create-and-install-package-version.ps1 index e73d415d0..0583ae0b6 100644 --- a/scripts/build/create-and-install-package-version.ps1 +++ b/scripts/build/create-and-install-package-version.ps1 @@ -127,7 +127,7 @@ function Install-Package-Version { ) $packageVersionId = "$packageVersionId".Trim() - npx sfdx force:package:install --noprompt --targetusername $targetusername --wait 20 --package $packageVersionId + npx sfdx force:package:install --noprompt --targetusername $targetusername --wait 20 --publishwait 5 --package $packageVersionId if ($LASTEXITCODE -ne 0) { throw "Error installing package version ID: $packageVersionId" } diff --git a/scripts/build/create-managed-package-beta-version.ps1 b/scripts/build/create-managed-package-beta-version.ps1 index 99aed9829..73b67cc93 100644 --- a/scripts/build/create-managed-package-beta-version.ps1 +++ b/scripts/build/create-managed-package-beta-version.ps1 @@ -12,7 +12,7 @@ Copy-Item -Path $rootProject -Destination $unlockedProject -Force Write-Output "Overwriting $rootProject with $managedProject" Copy-Item -Path $managedProject -Destination $rootProject -Force -# Create a beta of the managed package version (no `--codecoverage` flag) +# Create a new version of the managed package cp -R ./nebula-logger/core/ ./nebula-logger/managed-package/ $gitBranch = (git branch --show-current) $gitCommit = (git rev-parse HEAD) diff --git a/scripts/build/generate-apex-docs.ps1 b/scripts/build/generate-apex-docs.ps1 index 9af21dfd0..766fbe7f7 100644 --- a/scripts/build/generate-apex-docs.ps1 +++ b/scripts/build/generate-apex-docs.ps1 @@ -1,21 +1,17 @@ # This script is used to generate the markdown files used by Github pages for Apex docs -rm -f ./docs/apex/*/*.md -rm -rf ./docs/apex/configuration/ -rm -rf ./docs/apex/logger-engine/ -rm -rf ./docs/apex/log-management/ -rm -rf ./docs/apex/plugin-framework/ +find ./docs/apex/ -maxdepth 2 -type f -name "*.md" -delete -npx apexdocs-generate --configPath ./config/docs/apexdocs.json --scope global public --sourceDir ./nebula-logger/core/main/ --targetDir ./docs/apex --targetGenerator jekyll +npx apexdocs-generate --configPath ./config/docs/apexdocs.json --scope global public --sourceDir ./nebula-logger/core/ ./nebula-logger/plugins/ --targetDir ./docs/apex --targetGenerator jekyll # Make a few adjustments to the generated markdown files so that they work correctly in Github Pages $indexPageFile = "docs/apex/index.md" Write-Output "Processing file: $indexPageFile" (Get-Content -path $indexPageFile -Raw) -replace "# Classes","# Nebula Logger for Apex" | Set-Content -Path $indexPageFile -NoNewline (Get-Content -path $indexPageFile -Raw) -replace ".md","" | Set-Content -Path $indexPageFile -NoNewline -(Get-Content -path $indexPageFile -Raw) -replace "/Configuration/","configuration/" | Set-Content -Path $indexPageFile -NoNewline -(Get-Content -path $indexPageFile -Raw) -replace "/Logger-Engine/","logger-engine/" | Set-Content -Path $indexPageFile -NoNewline -(Get-Content -path $indexPageFile -Raw) -replace "/Log-Management/","log-management/" | Set-Content -Path $indexPageFile -NoNewline -(Get-Content -path $indexPageFile -Raw) -replace "/Plugin-Framework/","plugin-framework/" | Set-Content -Path $indexPageFile -NoNewline +(Get-Content -path $indexPageFile -Raw) -replace "/Configuration/","Configuration/" | Set-Content -Path $indexPageFile -NoNewline +(Get-Content -path $indexPageFile -Raw) -replace "/Logger-Engine/","Logger-Engine/" | Set-Content -Path $indexPageFile -NoNewline +(Get-Content -path $indexPageFile -Raw) -replace "/Log-Management/","Log-Management/" | Set-Content -Path $indexPageFile -NoNewline +(Get-Content -path $indexPageFile -Raw) -replace "/Plugins/","Plugins/" | Set-Content -Path $indexPageFile -NoNewline $docsSubdirectories = "docs/apex/*/*.*" foreach($file in Get-ChildItem $docsSubdirectories) { @@ -24,11 +20,9 @@ foreach($file in Get-ChildItem $docsSubdirectories) { (Get-Content -path $file -Raw) -replace "../Configuration/","" | Set-Content -Path $file -NoNewline (Get-Content -path $file -Raw) -replace "../Logger-Engine/","" | Set-Content -Path $file -NoNewline (Get-Content -path $file -Raw) -replace "../Log-Management/","" | Set-Content -Path $file -NoNewline - (Get-Content -path $file -Raw) -replace "../Plugin-Framework/","" | Set-Content -Path $file -NoNewline } -mv ./docs/apex/Configuration/ ./docs/apex/configuration/ -mv ./docs/apex/Logger-Engine/ ./docs/apex/logger-engine/ -mv ./docs/apex/Log-Management/ ./docs/apex/log-management/ -mv ./docs/apex/Plugin-Framework/ ./docs/apex/plugin-framework/ +# mv ./docs/apex/Configuration/ ./docs/apex/configuration/ +# mv ./docs/apex/Logger-Engine/ ./docs/apex/logger-engine/ +# mv ./docs/apex/Log-Management/ ./docs/apex/log-management/ prettier ./docs/apex/ --write \ No newline at end of file diff --git a/scripts/data/create-sample-log-entries.apex b/scripts/data/create-sample-log-entries.apex index 59446595d..052b81d65 100644 --- a/scripts/data/create-sample-log-entries.apex +++ b/scripts/data/create-sample-log-entries.apex @@ -1,5 +1,13 @@ +Logger.getUserSettings().DefaultSaveMethod__c = Logger.SaveMethod.EVENT_BUS.name(); +// Logger.getUserSettings().DefaultSaveMethod__c = 'BIG_OBJECT_IMMEDIATE'; +// Logger.getUserSettings().DefaultSaveMethod__c = 'BIG_OBJECT_QUEUEABLE'; +// Logger.getUserSettings().IsAnonymousModeEnabled__c = true; +// Logger.getUserSettings().DefaultPlatformEventStorageLocation__c = 'BIG_OBJECT'; +// Logger.getUserSettings().DefaultPlatformEventStorageLocation__c = 'CUSTOM_OBJECTS'; +upsert Logger.getUserSettings(); + Logger.setScenario('an example transaction scenario name'); -Logger.getUserSettings().LoggingLevel__c = LoggingLevel.INFO.name(); +Logger.getUserSettings().LoggingLevel__c = LoggingLevel.FINEST.name(); Logger.getUserSettings().IsDataMaskingEnabled__c = true; User currentUser = [SELECT Id, Name, Username, Profile.Name FROM User WHERE Id = :UserInfo.getUserId()]; @@ -12,9 +20,29 @@ currentUser.AboutMe = 'I hope you dont leak my social, which is 400-11-9999, btw Logger.error('Here is my fake Visa credit card 4000-1111-2222-0004, please don\'t steal it').addTag('data masking rule').addTag('credit card masking'); Logger.warn('Here is my fake Mastercard credit card 5000-1111-2222-0005, please don\'t steal it').addTag('data masking rule').addTag('credit card masking'); Logger.info('In case you want to steal my identity, my fake social is 400-11-9999, thanks', currentUser).addTag('data masking rule').addTag('an informational tag'); +Logger.debug('Here are some accounts', [SELECT Id, Name, Description FROM Account LIMIT 5]); +Logger.debug('Here are some save results', Database.update([SELECT Id, Name, Description FROM Account LIMIT 5], false)); +Logger.saveLog(); Logger.debug('Example DEBUG entry', currentUser); Logger.fine('Example FINE entry'); Logger.finer('Example FINER entry'); Logger.finest('Example FINEST entry'); +HttpRequest request = new HttpRequest(); +request.setBody('Hello, world! Here is my credit card number 5000-1111-2222-0005'); +request.setEndpoint('https://fake.salesforce.com'); +request.setMethod('GET'); +Logger.info('logging an HTTP request').setHttpRequestDetails(request); + + +HttpResponse response = new HttpResponse(); +response.setBody('Hello, world! Here is my credit card number 5000-1111-2222-0005'); +response.setHeader('someKey', 'some string value'); +response.setHeader('anotherKey', 'an amazing example value, wow'); +response.setStatus('STATUS_GOOD_JOB_YOU_DID_IT'); +response.setStatusCode(201); +Logger.info('logging an HTTP response').setHttpResponseDetails(response); + Logger.saveLog(); + +throw new DmlException('aahhhhh!'); diff --git a/scripts/data/create-sample-log-entry-archives.apex b/scripts/data/create-sample-log-entry-archives.apex new file mode 100644 index 000000000..8a6f98213 --- /dev/null +++ b/scripts/data/create-sample-log-entry-archives.apex @@ -0,0 +1,18 @@ +// Logger.getUserSettings().DefaultSaveMethod__c = 'BIG_OBJECT_EVENT_BUS'; +// Logger.getUserSettings().DefaultSaveMethod__c = 'BIG_OBJECT_IMMEDIATE'; +// Logger.getUserSettings().DefaultSaveMethod__c = 'BIG_OBJECT_QUEUEABLE'; +//Logger.getUserSettings().IsAnonymousModeEnabled__c = true; + +for (Integer i = 0; i < 200; i++) { + Logger.info('hello, LogEntryArchive__b! entry index==' + i).addTag('test tag'); +} +Logger.saveLog(); + +// try { +// Logger.debug('about to cause an exception').addTag('error logging'); +// update new User(); +// } catch(Exception ex) { +// Logger.error('got that exception that everyone was talking about, wild stuff', ex).addTag('error logging'); +// Logger.saveLog(); +// throw ex; +// } diff --git a/scripts/data/query-log-entry-archive-big-object.apex b/scripts/data/query-log-entry-archive-big-object.apex new file mode 100644 index 000000000..b0700f0d3 --- /dev/null +++ b/scripts/data/query-log-entry-archive-big-object.apex @@ -0,0 +1,15 @@ +Id userId = UserInfo.getUserId(); +List loggingLevelNames = new List(); +for (LoggingLevel logLevel : LoggingLevel.values()) { + loggingLevelNames.add(logLevel.name()); +} + +List archives = [ + SELECT Timestamp__c, LoggedBy__c, LoggingLevel__c, TransactionId__c, TransactionEntryNumber__c, Message__c, Tags__c + FROM LogEntryArchive__b + WHERE LoggedBy__c = :userId + AND LoggingLevel__c IN :loggingLevelNames + AND Timestamp__c >= LAST_WEEK + ORDER BY LoggedBy__c, LoggingLevel__c, Timestamp__c DESC +]; +System.debug('Matching LogEntryArchive__b records:\n' + JSON.serializePretty(archives)); \ No newline at end of file diff --git a/sfdx-project.json b/sfdx-project.json index c16879b92..1f150cf23 100644 --- a/sfdx-project.json +++ b/sfdx-project.json @@ -7,36 +7,62 @@ "package": "Nebula Logger - Core", "path": "./nebula-logger/core", "definitionFile": "./config/scratch-orgs/base-scratch-def.json", - "versionNumber": "4.7.0.NEXT", - "versionName": "Spring '22 Release", - "versionDescription": "Added new settings fields for controlling saving, default log scenario, & default log owner, improved logViewer lwc, bugfixes & cleanup", + "versionNumber": "4.7.1.NEXT", + "versionName": "Plugin Framework Overhaul", + "versionDescription": "New approach implemented for plugins, providing more control within LoggerSObjectHandler, and the ability to create plugins for LogBatchPurger", "releaseNotesUrl": "https://github.com/jongpie/NebulaLogger/releases", "default": true }, + { + "package": "Nebula Logger - Plugin - Big Object Archiving", + "path": "./nebula-logger/plugins/big-object-archiving/plugin", + "dependencies": [ + { + "package": "Nebula Logger - Core@4.7.1-8-plugin-framework-overhaul" + } + ], + "versionName": "Beta Release", + "versionNumber": "0.9.0.NEXT", + "versionDescription": "Initial beta version of new plugin", + "default": false + }, + { + "package": "Nebula Logger - Plugin - Log Retention Rules", + "path": "./nebula-logger/plugins/log-retention-rules/plugin", + "dependencies": [ + { + "package": "Nebula Logger - Core@4.7.1-8-plugin-framework-overhaul" + } + ], + "versionName": "Beta Release", + "versionNumber": "0.9.0.NEXT", + "versionDescription": "Initial beta version of new plugin", + "default": false + }, { "package": "Nebula Logger - Plugin - Logger Admin Dashboard", - "path": "./nebula-logger/plugins/Logger-Admin-Dashboard/plugin", + "path": "./nebula-logger/plugins/logger-admin-dashboard/plugin", "dependencies": [ { "package": "Nebula Logger - Core@4.6.12-0-log-scenario-rules" } ], "versionName": "Beta Release", - "versionNumber": "0.9.0.0", + "versionNumber": "0.9.0.NEXT", "versionDescription": "Initial beta version of new dashboard plugin", "default": false }, { "package": "Nebula Logger - Plugin - Slack", - "path": "./nebula-logger/plugins/Slack/plugin", + "path": "./nebula-logger/plugins/slack/plugin", "dependencies": [ { - "package": "Nebula Logger - Core@4.6.9-0-custom-metadata-types-optimized" + "package": "Nebula Logger - Core@4.7.1-8-plugin-framework-overhaul" } ], "versionName": "Beta Release: Round 3", - "versionNumber": "0.9.2.0", - "versionDescription": "Updated Slack plugin metadata to use Logger v4.6.9's renamed custom metadata type LoggerParameter__mdt", + "versionNumber": "0.10.0.NEXT", + "versionDescription": "TODO", "default": false }, { @@ -77,12 +103,17 @@ "Nebula Logger - Core@4.6.15-0-small-bugfixes-and-test-improvements": "04t5Y0000015lKDQAY", "Nebula Logger - Core@4.6.16-0-ui-cleanup": "04t5Y0000015lLzQAI", "Nebula Logger - Core@4.7.0-25-spring-'22-release": "04t5Y0000015lXSQAY", + "Nebula Logger - Core@4.7.1-8-plugin-framework-overhaul": "04t5Y0000015lgBQAQ", + "Nebula Logger - Plugin - Big Object Archiving": "0Ho5Y000000blMSSAY", + "Nebula Logger - Plugin - Big Object Archiving@0.9.0-2": "04t5Y0000015lgLQAQ", "Nebula Logger - Plugin - Log Retention Rules": "0Ho5Y000000blNfSAI", + "Nebula Logger - Plugin - Log Retention Rules@0.9.0-2": "04t5Y0000015lgGQAQ", "Nebula Logger - Plugin - Logger Admin Dashboard": "0Ho5Y000000blNkSAI", "Nebula Logger - Plugin - Logger Admin Dashboard@0.9.0-0": "04t5Y0000015l3yQAA", "Nebula Logger - Plugin - Slack": "0Ho5Y000000blMDSAY", "Nebula Logger - Plugin - Slack@0.9.0-0-beta-release": "04t5e00000061lHAAQ", "Nebula Logger - Plugin - Slack@0.9.1-0-beta-release-round-2": "04t5e00000065xiAAA", - "Nebula Logger - Plugin - Slack@0.9.2-0-beta-release-round-3": "04t5Y0000015l2WQAQ" + "Nebula Logger - Plugin - Slack@0.9.2-0-beta-release-round-3": "04t5Y0000015l2WQAQ", + "Nebula Logger - Plugin - Slack@0.10.0-2": "04t5Y0000015lgQQAQ" } }