package org.argeo.cms.internal.kernel; import static org.argeo.node.DataModelNamespace.CMS_DATA_MODEL_NAMESPACE; import java.io.InputStreamReader; import java.io.Reader; import java.lang.management.ManagementFactory; import java.net.URL; import java.util.HashSet; import java.util.Hashtable; import java.util.List; import java.util.Map; import java.util.Set; import javax.jcr.Repository; import javax.jcr.Session; import javax.security.auth.callback.CallbackHandler; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.apache.jackrabbit.commons.cnd.CndImporter; import org.apache.jackrabbit.core.RepositoryContext; import org.argeo.cms.CmsException; import org.argeo.jcr.JcrUtils; import org.argeo.node.DataModelNamespace; import org.argeo.node.NodeConstants; import org.argeo.node.NodeDeployment; import org.argeo.node.NodeState; import org.argeo.node.security.CryptoKeyring; import org.argeo.osgi.useradmin.UserAdminConf; import org.argeo.util.LangUtils; import org.osgi.framework.Bundle; import org.osgi.framework.BundleContext; import org.osgi.framework.Constants; import org.osgi.framework.FrameworkUtil; import org.osgi.framework.ServiceReference; import org.osgi.framework.wiring.BundleCapability; import org.osgi.framework.wiring.BundleWire; import org.osgi.framework.wiring.BundleWiring; import org.osgi.service.cm.Configuration; import org.osgi.service.cm.ConfigurationAdmin; import org.osgi.service.cm.ManagedService; import org.osgi.service.useradmin.UserAdmin; import org.osgi.util.tracker.ServiceTracker; public class CmsDeployment implements NodeDeployment { private final static String LEGACY_JCR_REPOSITORY_ALIAS = "argeo.jcr.repository.alias"; private final Log log = LogFactory.getLog(getClass()); private final BundleContext bc = FrameworkUtil.getBundle(getClass()).getBundleContext(); private DeployConfig deployConfig; private HomeRepository homeRepository; private Long availableSince; private final boolean cleanState; private NodeHttp nodeHttp; // Readiness private boolean nodeAvailable = false; private boolean userAdminAvailable = false; private boolean httpExpected = false; private boolean httpAvailable = false; public CmsDeployment() { ServiceReference nodeStateSr = bc.getServiceReference(NodeState.class); if (nodeStateSr == null) throw new CmsException("No node state available"); NodeState nodeState = bc.getService(nodeStateSr); cleanState = nodeState.isClean(); nodeHttp = new NodeHttp(); initTrackers(); } private void initTrackers() { ServiceTracker httpSt = new ServiceTracker(bc, NodeHttp.class, null) { @Override public NodeHttp addingService(ServiceReference reference) { httpAvailable = true; checkReadiness(); return super.addingService(reference); } }; // httpSt.open(); KernelUtils.asyncOpen(httpSt); ServiceTracker repoContextSt = new RepositoryContextStc(); // repoContextSt.open(); KernelUtils.asyncOpen(repoContextSt); ServiceTracker userAdminSt = new ServiceTracker(bc, UserAdmin.class, null) { @Override public UserAdmin addingService(ServiceReference reference) { userAdminAvailable = true; checkReadiness(); return super.addingService(reference); } }; // userAdminSt.open(); KernelUtils.asyncOpen(userAdminSt); ServiceTracker confAdminSt = new ServiceTracker(bc, ConfigurationAdmin.class, null) { @Override public ConfigurationAdmin addingService(ServiceReference reference) { ConfigurationAdmin configurationAdmin = bc.getService(reference); deployConfig = new DeployConfig(configurationAdmin, cleanState); httpExpected = deployConfig.getProps(KernelConstants.JETTY_FACTORY_PID, "default") != null; try { // Configuration[] configs = configurationAdmin // .listConfigurations("(service.factoryPid=" + // NodeConstants.NODE_REPOS_FACTORY_PID + ")"); // for (Configuration config : configs) { // Object cn = config.getProperties().get(NodeConstants.CN); // if (log.isDebugEnabled()) // log.debug("Standalone repo cn: " + cn); // } Configuration[] configs = configurationAdmin .listConfigurations("(service.factoryPid=" + NodeConstants.NODE_USER_ADMIN_PID + ")"); boolean hasDomain = false; for (Configuration config : configs) { Object realm = config.getProperties().get(UserAdminConf.realm.name()); if (realm != null) { log.debug("Found realm: " + realm); hasDomain = true; } } if (hasDomain) { loadIpaJaasConfiguration(); } } catch (Exception e) { throw new CmsException("Cannot initialize config", e); } return super.addingService(reference); } }; // confAdminSt.open(); KernelUtils.asyncOpen(confAdminSt); } private void loadIpaJaasConfiguration() { if (System.getProperty(KernelConstants.JAAS_CONFIG_PROP) == null) { String jaasConfig = KernelConstants.JAAS_CONFIG_IPA; URL url = getClass().getClassLoader().getResource(jaasConfig); KernelUtils.setJaasConfiguration(url); log.debug("Set IPA JAAS configuration."); } } public void shutdown() { if (nodeHttp != null) nodeHttp.destroy(); if (deployConfig != null) deployConfig.save(); } private void checkReadiness() { if (nodeAvailable && userAdminAvailable && (httpExpected ? httpAvailable : true)) { String data = KernelUtils.getFrameworkProp(KernelUtils.OSGI_INSTANCE_AREA); String state = KernelUtils.getFrameworkProp(KernelUtils.OSGI_CONFIGURATION_AREA); availableSince = System.currentTimeMillis(); long jvmUptime = ManagementFactory.getRuntimeMXBean().getUptime(); String jvmUptimeStr = " in " + (jvmUptime / 1000) + "." + (jvmUptime % 1000) + "s"; log.info("## ARGEO NODE AVAILABLE" + (log.isDebugEnabled() ? jvmUptimeStr : "") + " ##"); if (log.isDebugEnabled()) { log.debug("## state: " + state); if (data != null) log.debug("## data: " + data); } long begin = bc.getService(bc.getServiceReference(NodeState.class)).getAvailableSince(); long initDuration = System.currentTimeMillis() - begin; if (log.isTraceEnabled()) log.trace("Kernel initialization took " + initDuration + "ms"); tributeToFreeSoftware(initDuration); } } final private void tributeToFreeSoftware(long initDuration) { if (log.isTraceEnabled()) { long ms = initDuration / 100; log.trace("Spend " + ms + "ms" + " reflecting on the progress brought to mankind" + " by Free Software..."); long beginNano = System.nanoTime(); try { Thread.sleep(ms, 0); } catch (InterruptedException e) { // silent } long durationNano = System.nanoTime() - beginNano; final double M = 1000d * 1000d; double sleepAccuracy = ((double) durationNano) / (ms * M); log.trace("Sleep accuracy: " + String.format("%.2f", 100 - (sleepAccuracy * 100 - 100)) + " %"); } } private void prepareNodeRepository(Repository deployedNodeRepository) { if (availableSince != null) { throw new CmsException("Deployment is already available"); } // home prepareDataModel(KernelUtils.openAdminSession(deployedNodeRepository)); } private void prepareHomeRepository(Repository deployedRepository) { Hashtable regProps = new Hashtable(); regProps.put(NodeConstants.CN, NodeConstants.HOME); regProps.put(LEGACY_JCR_REPOSITORY_ALIAS, NodeConstants.HOME); homeRepository = new HomeRepository(deployedRepository); // register bc.registerService(Repository.class, homeRepository, regProps); new ServiceTracker(bc, CallbackHandler.class, null) { @Override public CallbackHandler addingService(ServiceReference reference) { NodeKeyRing nodeKeyring = new NodeKeyRing(homeRepository); CallbackHandler callbackHandler = bc.getService(reference); nodeKeyring.setDefaultCallbackHandler(callbackHandler); bc.registerService(LangUtils.names(CryptoKeyring.class, ManagedService.class), nodeKeyring, LangUtils.dico(Constants.SERVICE_PID, NodeConstants.NODE_KEYRING_PID)); return callbackHandler; } }.open(); } /** Session is logged out. */ private void prepareDataModel(Session adminSession) { try { Set processed = new HashSet(); bundles: for (Bundle bundle : bc.getBundles()) { BundleWiring wiring = bundle.adapt(BundleWiring.class); if (wiring == null) continue bundles; processWiring(adminSession, wiring, processed); } } finally { JcrUtils.logoutQuietly(adminSession); } } private void processWiring(Session adminSession, BundleWiring wiring, Set processed) { // recursively process requirements first List requiredWires = wiring.getRequiredWires(CMS_DATA_MODEL_NAMESPACE); for (BundleWire wire : requiredWires) { processWiring(adminSession, wire.getProviderWiring(), processed); // registerCnd(adminSession, wire.getCapability(), processed); } List capabilities = wiring.getCapabilities(CMS_DATA_MODEL_NAMESPACE); for (BundleCapability capability : capabilities) { registerDataModelCapability(adminSession, capability, processed); } } private void registerDataModelCapability(Session adminSession, BundleCapability capability, Set processed) { Map attrs = capability.getAttributes(); String name = (String) attrs.get(DataModelNamespace.CAPABILITY_NAME_ATTRIBUTE); if (processed.contains(name)) { if (log.isTraceEnabled()) log.trace("Data model " + name + " has already been processed"); return; } // CND String path = (String) attrs.get(DataModelNamespace.CAPABILITY_CND_ATTRIBUTE); if (path != null) { URL url = capability.getRevision().getBundle().getResource(path); if (url == null) throw new CmsException("No data model '" + name + "' found under path " + path); try (Reader reader = new InputStreamReader(url.openStream())) { CndImporter.registerNodeTypes(reader, adminSession, true); processed.add(name); if (log.isDebugEnabled()) log.debug("Registered CND " + url); } catch (Exception e) { throw new CmsException("Cannot import CND " + url, e); } } if (!asBoolean((String) attrs.get(DataModelNamespace.CAPABILITY_ABSTRACT_ATTRIBUTE))) { Hashtable properties = new Hashtable<>(); properties.put(LEGACY_JCR_REPOSITORY_ALIAS, name); properties.put(NodeConstants.CN, name); if (name.equals(NodeConstants.NODE)) properties.put(Constants.SERVICE_RANKING, Integer.MAX_VALUE); LocalRepository localRepository = new LocalRepository(adminSession.getRepository(), capability); bc.registerService(Repository.class, localRepository, properties); if (log.isDebugEnabled()) log.debug("Published data model " + name); } } private boolean asBoolean(String value) { if (value == null) return false; switch (value) { case "true": return true; case "false": return false; default: throw new CmsException("Unsupported value for attribute " + DataModelNamespace.CAPABILITY_ABSTRACT_ATTRIBUTE + ": " + value); } } @Override public Long getAvailableSince() { return availableSince; } private class RepositoryContextStc extends ServiceTracker { public RepositoryContextStc() { super(bc, RepositoryContext.class, null); } @Override public RepositoryContext addingService(ServiceReference reference) { RepositoryContext nodeRepo = bc.getService(reference); Object cn = reference.getProperty(NodeConstants.CN); if (cn != null) { if (cn.equals(NodeConstants.NODE)) { prepareNodeRepository(nodeRepo.getRepository()); prepareHomeRepository(nodeRepo.getRepository()); nodeAvailable = true; checkReadiness(); } else { // TODO standalone } } return nodeRepo; } @Override public void modifiedService(ServiceReference reference, RepositoryContext service) { } @Override public void removedService(ServiceReference reference, RepositoryContext service) { } } }