Settings Enforcer Utility
What the Settings Enforcer Utility module adds to your project
Settings Enforcer allows projects to externalize configuration values in Sites & Settings and Singletons into a properties file or context.xml via the Settings API.
Settings that are “forced” in this way are no longer allowed to be edited in the CMS. If the value is set to null (an empty string), the field on the Sites & Settings page will also be hidden. If the value is set to something other than null, the field will be read only.
Sites & Settings values that should be the same across all environments (production, lower environments, local dev) can be set in the project’s settings.properties
using this syntax:
force/global/enableTrailingSlashNormalization=true
# RTE settings we don't want editors to change
force/global/legacyHtml=
# 4.2
force/global/enableCodeMirrorRichTextEditor=
# 4.1
force/global/enableProseMirrorRichTextEditor=true
# Collections
force/global/listField/0=first
force/global/listField/1=second
Settings that should be different across environments can be set in each environment’s context.xml
file using this syntax:
<Environment name="force/global/express.cmsToolModification.defaultTaskHost" value="task.brightspot.service.${brightspot.environment}.my-project.internal" type="java.lang.String"/>
Installation
api('com.psddev.component-lib:util-settings-enforcer')
Programmatically Detecting Whether a Setting is Enforced
Sometimes it is useful to know whether a setting is enforced. For example, when synchronizing Site settings in Stage Preview, any setting that is enforced should not be part of the calculation to determine whether the field values are in sync. If different servers have different enforced settings, copying enforced settings from one server to the other will result in an infinite loop! In order to avoid this, the SettingsEnforcer Modification<SiteSettings>
has a getFields()
method that returns all of the internal field names of settings that have been enforced on the Site or CmsTool.
CmsTool tool = Query.from(CmsTool.class).first();
if (tool.as(SiteSettingsEnforcer.class).getFields().contains("fieldInternalName")) {
// fieldInternalName is handled via Settings Enforcer; do not modify!
}
Scope and Syntax
Each setting name has multiple parts separated by slashes:
- The
force
prefix - One of either
global
,all
,sites
,site
, or the fully qualified class name of any Singleton. - If (2) is
site
, the name of a site or group of sites. This can be any string you want, and will populate the “Enforced Settings” dropdown on Site. Otherwise skip to (4). - The
internalName
of the setting you wish to force. - Optional for collections only - an integer value indicating the list item order. For example /0, /1, /2.
Examples:
-
force/global/defaultPreviewUrl
will set thedefaultPreviewUrl
on the CmsTool singleton. -
force/all/broadcastMessage
will set thebroadcastMessage
on the CmsTool singleton as well as all Sites. -
force/sites/allowSiteCopy
will setallowSiteCopy
on all Sites, but not the CmsTool singleton. -
force/site/groupOne/cmsCssClass
will set thecmsCssClass
on all Sites that have selected “Group One” on the “Enforced Settings” dropdown in the “Advanced” cluster. Note that settings enforced at a per-site level like this will not be hidden or read only unless a default is supplied inglobal
,all
, orsites
. -
force/com.psddev.migration.MigrationTool/taskHost
will set thetaskHost
of the MigrationTool Singleton. -
force/global/debuggingToEmails/0
andforce/global/debuggingToEmails/1
will setdebuggingToEmails
list to a list of 2 values on the CmsTool singleton.
Adding the Settings Enforcer Utility module to your project
Follow these steps:
- Add the package dependency to the appropriate
pom.xml
files. Using a released version is advised.<dependency> <groupId>com.psddev.component-lib</groupId> <artifactId>util-settings-enforcer</artifactId> </dependency>
or build.gradle
:
api 'com.psddev.component-lib:util-settings-enforcer'
- Create or modify a
settings.properties
file in your project.
A common location for this file is ./site/src/main/resources/settings.properties
Known Limitations
Only boolean, string, numbers, and collections of same are supported.
Examples
Here are several common settings that should be enforced.
The following settings can be copied into your project as ./site/src/main/resources/settings.properties
:
# Deprecated / unnecessary / dangerous settings that users shouldn't touch
force/global/allowInsecureAuthenticationCookie=
force/global/alwaysGeneratePermalinks=
force/global/disableAutomaticallySavingDrafts=
force/global/disableEditFieldUpdateCache=
force/global/disableFrontEndUploader=
force/global/disableInvisibleContentPreview=
force/global/disableRtc=
force/global/disableSmartUploader=
force/global/disableToolChecks=
force/global/disableViewers=
force/global/disableWorkInProgress=
force/global/disabledPlugins/0=developerBackgroundTasks
force/global/disabledPlugins/1=developerBulkOperations
force/global/disabledPlugins/2=developerSqlDebug
force/global/displayTypesNotAssociatedWithJavaClasses=
force/global/enableAjaxSaves=
force/global/enableAnnotations=
force/global/enablePaddedCrop=
force/global/legacyHtml=
force/global/singleGeneratedPermalink=
force/global/useLegacySettingsSiteSwitching=
force/global/useNonMinifiedCss=
force/global/useNonMinifiedJavaScript=
force/global/useOldDirectoryPathsPredicate=
force/global/useOldHistoryIndex=
force/global/useOldTaxonomyChildrenDetection=
force/com.psddev.migration.MigrationTool/url=
# Whole-project defaults - review these and decide per project
force/global/disableVideoEditorUI=
force/global/displaySiteInSearchResult=true
force/global/enableCrossDomainInlineEditing=true
force/global/enableTrailingSlashNormalization=true
force/global/express.cmsToolModification.autoIncrementPermalinks=
force/global/removeTrailingSlashes=
force/global/searchCarouselDisplay=DISABLED
force/global/useLegacySearchWithContent=
# Disallow Global as a site
force/global/allowGlobalEditing=
force/global/defaultSiteUrl=
force/global/siteMapDefaultUrl=
force/global/sitemap.generateSiteMapTypeClassNames=
force/global/url=
# RTE
force/global/enableCodeMirrorRichTextEditor=
force/global/enableProseMirrorRichTextEditor=true
# Disallow Content Locking
force/global/allowDisableUnlocking=
force/global/contentLocking=
force/global/hardLockTimeout=
force/global/inactivityLockTimeout=
force/global/networkLockTimeout=
These settings should be configured per environment in context.xml
.
- Substitute instances of
MY-PROJECT
with appropriate values - Substitute value for
TASK-HOST
from the following options:- Normal task host - all environments (prod, qa, uat) (legacy infrastructure):
task.brightspot.service.${brightspot.environment}.MY-PROJECT.internal
- Dedicated job host - production only (legacy infrastructure):
master.job.service.${brightspot.environment}.MY-PROJECT.internal
- Dedicated job host - production only (cloud infrastructure):
brightspot-task.MY-PROJECT.svc.cluster.local
- Brightspot Docker:
${brightspot.hostname}
- Normal task host - all environments (prod, qa, uat) (legacy infrastructure):
- Substitute
https://cms.example.com/
with your CMS URL. - Find the appropriate values for all AWS integration settings - many of them will be different per environment!
- Any AWS setting that is enforced in production should also be enforced in lower and local environments. This will avoid accidentally connecting to the production AWS service after doing a production copydown!
- Note that you may or may not need all of these options based on your project requirements.
<!-- Task Hosts -->
<Environment name="force/global/express.cmsToolModification.defaultTaskHost" value="TASK-HOST" type="java.lang.String" />
<Environment name="force/global/analytics.pageViews.taskHost" value="TASK-HOST" type="java.lang.String" />
<Environment name="force/global/google.taskHost" value="TASK-HOST" type="java.lang.String" />
<Environment name="force/com.psddev.migration.MigrationTool/taskHost" value="TASK-HOST" type="java.lang.String" />
<!-- Per environment settings -->
<Environment name="force/global/environment" value="${brightspot.environment}" type="java.lang.String" />
<Environment name="force/global/defaultPreviewUrl" value="https://cms.example.com/" type="java.lang.String" />
<Environment name="force/global/defaultToolUrl" value="https://cms.example.com/" type="java.lang.String" />
<!-- Mail Publishing -->
<Environment name="force/global/s3Region" value="" type="java.lang.String" />
<Environment name="force/global/queueName" value="" type="java.lang.String" />
<!-- AWS integration settings -->
<Environment name="force/global/aws.ses.enableNotificationService" value="true" type="java.lang.String" />
<Environment name="force/global/aws.ses.bounceNotificationQueueName" value="" type="java.lang.String" />
<Environment name="force/global/aws.ses.complaintNotificationQueueName" value="" type="java.lang.String" />
<Environment name="force/global/aws.sqs.region" value="" type="java.lang.String" />
<Environment name="force/global/aws.comprehend.region" value="" type="java.lang.String" />
<Environment name="force/global/bsp.elemental.region" value="" type="java.lang.String" />
<Environment name="force/global/bsp.elemental.mediaConvertRole" value="" type="java.lang.String" />
<Environment name="force/global/bsp.elemental.mediaConvertQueueName" value="" type="java.lang.String" />
<Environment name="force/global/bsp.elemental.mediaLiveRole" value="" type="java.lang.String" />
<Environment name="force/global/aws.rekognition.region" value="" type="java.lang.String" />
<Environment name="force/global/rekognitionVideo.queueName" value="" type="java.lang.String" />
<Environment name="force/global/rekognitionVideo.topicArn" value="" type="java.lang.String" />
<Environment name="force/global/rekognitionVideo.roleArn" value="" type="java.lang.String" />
<Environment name="force/global/aws.textract.queueName" value="" type="java.lang.String" />
<Environment name="force/global/aws.textract.topicArn" value="" type="java.lang.String" />
<Environment name="force/global/aws.textract.roleArn" value="" type="java.lang.String" />
<Environment name="force/global/aws.transcribe.queueName" value="" type="java.lang.String" />
<Environment name="force/global/aws.transcribe.region" value="" type="java.lang.String" />