Read the Adobe provided documentation for Conf and ConfMgr.
I have created a demo project and hosted it on GitHub at https://github.com/nateyolles/aem-slash-conf.
As an example, let's host a multi-tenant AEM instance with three projects and one subproject: Aviato, Pied Piper, Hooli, and subproject Hooli XYZ.
/content/aviato /content/piedpiper /content/hooli /content/hooli/hoolixyz
Add a cq:conf property of type String at the root of every project page and point it to the site's corresponding configuration path. A global configuration will be used if a cq:conf property is not specified as is the case for Aviato in this example.
/content/piedpiper/jcr:content/@cq:conf = "/conf/tentants/piedpiper" /content/hooli/jcr:content/@cq:conf = "/conf/tentants/hooli" /content/hooli/hoolixyz/jcr:content/@cq:conf = "/conf/tentants/hooli/hoolixyz"
Conf respects relative paths as it pertains to the cq:conf property. Therefore, the following settings could also be used:
/content/hooli/jcr:content/@cq:conf = "/conf/tentants/hooli" /content/hooli/hoolixyz/jcr:content/@cq:conf = "hoolixyz"
The tree structure of /conf should closely resemble /content. The configuration nodes should be of type cq:Page with the configuration properties on the jcr:content sub-nodes. All configurations start below the path specified in cq:conf with the addition of /settings.
/conf/global/settings/socialmedia/facebook/jcr:content/@enabled = false /conf/tenants/piedpiper/settings/socialmedia/facebook/jcr:content/@enabled = true /conf/tenants/hooli/settings/socialmedia/facebook/jcr:content/@enabled = true /conf/tenants/hooli/hoolixyz/settings/socialmedia/facebook/jcr:content/@enabled = true
/conf/global is special as it provides an ultimate fallback in the Conf inheritance mechanism. It this example Aviato does not have its own config, so it would use the global configs. /conf/tenants are not dictated by AEM but is a convention worth following to help prevent namespace collision in multi-tenant scenarios.
Configs can live under /conf, /apps and /libs with the same apps-then-libs type resolution that you are already familiar with. The resolution order can be set in the Felix console com.adobe.granite.confmgr.impl.ConfMgrImpl configuration: http://localhost:4502/system/console/configMgr/com.adobe.granite.confmgr.impl.ConfMgrImpl.
Use a Conf object to get a site configuration. The Conf object is usually obtained by adapted a Resource or Page using Apache Sling's adaptTo() method. Once you have the site's Conf object, you can get any number of predefined ValueMaps and properties. Notice that the ValueMaps cannot be modified – they are read-only. Properties are retrieved from the ValueMap in the familiar way of passing in either a default value or the class type.
The Conf object you retrieve is based on the path of the resource. So if have a Sling Model or WCMUsePojo and you adapt the current page, you will get a different Conf than if you adapt the current resource.
ConfModel.java
package com.nateyolles.aem.slashconf.core.models;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.models.annotations.Model;
import com.adobe.granite.confmgr.Conf;
import com.day.cq.wcm.api.Page;
/**
* AEM com.adobe.granite.confmgr.Conf example of getting a configuration by
* using adaptTo and reading various settings and defining default values.
*/
@Model(adaptables = SlingHttpServletRequest.class)
public class ConfModel {
@Inject
private Page currentPage;
@PostConstruct
protected void init() {
final Conf conf = currentPage.adaptTo(Conf.class);
final ValueMap fooSettings = conf.getItem("socialmedia/foo");
final String fooName = fooSettings.get("name", String.class);
final String fooNameAlt = fooSettings.get("nameAlt", "unavailable");
final boolean fooEnabled = fooSettings.get("enabled", false);
final long limit = fooSettings.get("limit", 25);
}
}
As you can see, the ConfMgr is a fairly lightweight and transparent helper. Everything is in the JCR, so configurations can be accessed through normal channels such as a Resource Resolver. However, there are three main benefits to using com.adobe.granite.confmgr.Conf.
The most obvious advantage is the built-in inheritance management. For instance, the site for Hooli might have ten configurations. If the subproject Hooli XYZ overwrites only one of those configurations, it will use the other nine provided by Hooli. If the Conf object is looking for a config that doesn't exist in the subproject or project, it will look under /global. Conf inherits and overwrites at the node level, not the property level.
Conf handles null values for you. If the configuration, ValueMap, or individual properties are null, you can continue without null checks and you won't get a Null Pointer Exception.
Conf respects ACLs and multi-tenant privacy. If you set up your ACLs correctly, a shared component between two tenants will still only be able to access the configurations of the current tenant. Conf does not allow relative paths that try to move up the tree and into another tenant (i.e. "../../differentSite/foo/bar").
ConMgr can be used as a service.
ConfServiceModel.java
package com.nateyolles.aem.slashconf.core.models;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.models.annotations.Model;
import com.adobe.granite.confmgr.Conf;
import com.adobe.granite.confmgr.ConfMgr;
/**
* AEM com.adobe.granite.confmgr.Conf example of getting a configuration using
* the ConfMgr service rather than adapting a Resource.
*/
@Model(adaptables = SlingHttpServletRequest.class)
public class ConfServiceModel {
@Inject
private ConfMgr confMgr;
@Inject
private Resource currentResource;
@PostConstruct
protected void init() {
final Conf conf = confMgr.getConf(currentResource);
final ValueMap fooSettings = conf.getItem("socialmedia/foo");
final String name = fooSettings.get("name", String.class);
}
}
When used as a service, a Resource Resolver can be passed in which allows the use of Sling Service Accounts, thereby respecting ACLs on the configurations.
ConfServiceAccountModel.java
package com.nateyolles.aem.slashconf.core.models;
import java.util.HashMap;
import java.util.Map;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.LoginException;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ResourceResolver;
import org.apache.sling.api.resource.ResourceResolverFactory;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.models.annotations.Model;
import com.adobe.granite.confmgr.Conf;
import com.adobe.granite.confmgr.ConfMgr;
import com.day.cq.wcm.api.Page;
/**
* AEM com.adobe.granite.confmgr.Conf example of getting a configuration while
* respecting permissions by using a Resource Resolver and a Service Account.
*/
@Model(adaptables = SlingHttpServletRequest.class)
public class ConfServiceAccountModel {
@Inject
private Page currentPage;
@Inject
private ConfMgr confMgr;
@Inject
private ResourceResolverFactory resolverFactory;
@PostConstruct
protected void init() {
final Map<String, Object> serviceParams = new HashMap<String, Object>();
serviceParams.put(ResourceResolverFactory.SUBSERVICE, "readService");
ResourceResolver serviceResolver = null;
try {
serviceResolver = resolverFactory.getServiceResourceResolver(serviceParams);
final Conf conf = confMgr.getConf(currentPage.adaptTo(Resource.class), serviceResolver);
final ValueMap fooSettings = conf.getItem("socialmedia/foo");
final String name = fooSettings.get("name", String.class);
} catch (LoginException e) {
// TODO Auto-generated catch block
} finally {
if (serviceResolver != null && serviceResolver.isLive()) {
serviceResolver.close();
serviceResolver = null;
}
}
}
}
All inherited configurations can be obtained by using the Conf#getList() method, which returns a List of ValueMaps. Configurations can also be chosen explicitly by using a Resource Resolver to get the specific resource.
ConfOtherModel.java
package com.nateyolles.aem.slashconf.core.models;
import java.util.List;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.ResourceResolver;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.models.annotations.Model;
import com.adobe.granite.confmgr.Conf;
/**
* AEM com.adobe.granite.confmgr.Conf examples of getting a specific
* configuration using a Resource Resolver as well as getting a List inherited
* configurations using the getList method.
*/
@Model(adaptables = SlingHttpServletRequest.class)
public class ConfOtherModel {
@Inject
ResourceResolver resolver;
@PostConstruct
protected void init() {
/* Get a specific configuration */
final Conf conf = resolver.getResource("/conf/tenants/hooli/hoolixyz").adaptTo(Conf.class);
final ValueMap fooSettings = conf.getItem("socialmedia/foo");
final String fooName = fooSettings.get("name", String.class);
/* Get a list of configurations including all confs in the inheritance tree. */
final List<ValueMap> fooSettingsList = conf.getList("socialmedia/foo");
for (ValueMap properties : fooSettingsList) {
properties.get("name", String.class);
}
}
}
View a complete example at https://github.com/nateyolles/aem-slash-conf.
Thanks to Alex Klimetschek ( @alexkli) for sharing his knowledge of /conf of which much of this blog post is based on.
package com.nateyolles.aem.slashconf.core.models;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.models.annotations.Model;
import com.adobe.granite.confmgr.Conf;
import com.adobe.granite.confmgr.ConfMgr;
/**
* AEM com.adobe.granite.confmgr.Conf example of getting a configuration using
* the ConfMgr service rather than adapting a Resource.
*/
@Model(adaptables = SlingHttpServletRequest.class)
public class ConfServiceModel {
@Inject
private ConfMgr confMgr;
@Inject
private Resource currentResource;
@PostConstruct
protected void init() {
final Conf conf = confMgr.getConf(currentResource);
final ValueMap fooSettings = conf.getItem("socialmedia/foo");
final String name = fooSettings.get("name", String.class);
}
}
When used as a service, a Resource Resolver can be passed in which allows the use of Sling Service Accounts, thereby respecting ACLs on the configurations.
ConfServiceAccountModel.java
package com.nateyolles.aem.slashconf.core.models;
import java.util.HashMap;
import java.util.Map;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.LoginException;
import org.apache.sling.api.resource.Resource;
import org.apache.sling.api.resource.ResourceResolver;
import org.apache.sling.api.resource.ResourceResolverFactory;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.models.annotations.Model;
import com.adobe.granite.confmgr.Conf;
import com.adobe.granite.confmgr.ConfMgr;
import com.day.cq.wcm.api.Page;
/**
* AEM com.adobe.granite.confmgr.Conf example of getting a configuration while
* respecting permissions by using a Resource Resolver and a Service Account.
*/
@Model(adaptables = SlingHttpServletRequest.class)
public class ConfServiceAccountModel {
@Inject
private Page currentPage;
@Inject
private ConfMgr confMgr;
@Inject
private ResourceResolverFactory resolverFactory;
@PostConstruct
protected void init() {
final Map<String, Object> serviceParams = new HashMap<String, Object>();
serviceParams.put(ResourceResolverFactory.SUBSERVICE, "readService");
ResourceResolver serviceResolver = null;
try {
serviceResolver = resolverFactory.getServiceResourceResolver(serviceParams);
final Conf conf = confMgr.getConf(currentPage.adaptTo(Resource.class), serviceResolver);
final ValueMap fooSettings = conf.getItem("socialmedia/foo");
final String name = fooSettings.get("name", String.class);
} catch (LoginException e) {
// TODO Auto-generated catch block
} finally {
if (serviceResolver != null && serviceResolver.isLive()) {
serviceResolver.close();
serviceResolver = null;
}
}
}
}
All inherited configurations can be obtained by using the Conf#getList() method, which returns a List of ValueMaps. Configurations can also be chosen explicitly by using a Resource Resolver to get the specific resource.
ConfOtherModel.java
package com.nateyolles.aem.slashconf.core.models;
import java.util.List;
import javax.annotation.PostConstruct;
import javax.inject.Inject;
import org.apache.sling.api.SlingHttpServletRequest;
import org.apache.sling.api.resource.ResourceResolver;
import org.apache.sling.api.resource.ValueMap;
import org.apache.sling.models.annotations.Model;
import com.adobe.granite.confmgr.Conf;
/**
* AEM com.adobe.granite.confmgr.Conf examples of getting a specific
* configuration using a Resource Resolver as well as getting a List inherited
* configurations using the getList method.
*/
@Model(adaptables = SlingHttpServletRequest.class)
public class ConfOtherModel {
@Inject
ResourceResolver resolver;
@PostConstruct
protected void init() {
/* Get a specific configuration */
final Conf conf = resolver.getResource("/conf/tenants/hooli/hoolixyz").adaptTo(Conf.class);
final ValueMap fooSettings = conf.getItem("socialmedia/foo");
final String fooName = fooSettings.get("name", String.class);
/* Get a list of configurations including all confs in the inheritance tree. */
final List<ValueMap> fooSettingsList = conf.getList("socialmedia/foo");
for (ValueMap properties : fooSettingsList) {
properties.get("name", String.class);
}
}
}
View a complete example at https://github.com/nateyolles/aem-slash-conf.
Thanks to Alex Klimetschek ( @alexkli) for sharing his knowledge of /conf of which much of this blog post is based on.
No comments:
Post a Comment
If you have any doubts or questions, please let us know.