/** * Copyright (c) 2014-2017 by the respective copyright holders. * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html */ package org.eclipse.smarthome.core.binding.xml.internal; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; import java.util.LinkedHashSet; import java.util.List; import java.util.Locale; import java.util.Map; import java.util.Map.Entry; import java.util.Set; import org.eclipse.smarthome.config.core.ConfigDescriptionProvider; import org.eclipse.smarthome.config.xml.AbstractXmlConfigDescriptionProvider; import org.eclipse.smarthome.config.xml.osgi.XmlDocumentBundleTracker; import org.eclipse.smarthome.config.xml.osgi.XmlDocumentProviderFactory; import org.eclipse.smarthome.config.xml.util.XmlDocumentReader; import org.eclipse.smarthome.core.binding.BindingInfo; import org.eclipse.smarthome.core.binding.BindingInfoProvider; import org.eclipse.smarthome.core.i18n.BindingI18nUtil; import org.eclipse.smarthome.core.i18n.I18nProvider; import org.osgi.framework.Bundle; import org.osgi.service.component.ComponentContext; import org.osgi.service.component.annotations.Activate; import org.osgi.service.component.annotations.Component; import org.osgi.service.component.annotations.Deactivate; import org.osgi.service.component.annotations.Reference; /** * The {@link XmlBindingInfoProvider} is a concrete implementation of the {@link BindingInfoProvider} service interface. * <p> * This implementation manages any {@link BindingInfo} objects associated to specific modules. If a specific module * disappears, any registered {@link BindingInfo} objects associated with that module are released. * * @author Michael Grammling - Initial Contribution * @author Michael Grammling - Refactoring: Provider/Registry pattern is used, added locale support */ @Component public class XmlBindingInfoProvider implements BindingInfoProvider { private static final String XML_DIRECTORY = "/ESH-INF/binding/"; private Map<Bundle, List<BindingInfo>> bundleBindingInfoMap; private BindingI18nUtil bindingI18nUtil; private AbstractXmlConfigDescriptionProvider configDescriptionProvider; private XmlDocumentBundleTracker<BindingInfoXmlResult> bindingInfoTracker; public XmlBindingInfoProvider() { this.bundleBindingInfoMap = new HashMap<>(10); } @Activate public void activate(ComponentContext componentContext) { XmlDocumentReader<BindingInfoXmlResult> bindingInfoReader = new BindingInfoReader(); XmlDocumentProviderFactory<BindingInfoXmlResult> bindingInfoProviderFactory = new BindingInfoXmlProviderFactory( this, configDescriptionProvider); bindingInfoTracker = new XmlDocumentBundleTracker<>(componentContext.getBundleContext(), XML_DIRECTORY, bindingInfoReader, bindingInfoProviderFactory); bindingInfoTracker.open(); } @Deactivate public void deactivate(ComponentContext componentContext) { bindingInfoTracker.close(); bindingInfoTracker = null; } private List<BindingInfo> acquireBindingInfos(Bundle bundle) { if (bundle != null) { List<BindingInfo> bindingInfos = this.bundleBindingInfoMap.get(bundle); if (bindingInfos == null) { bindingInfos = new ArrayList<BindingInfo>(10); this.bundleBindingInfoMap.put(bundle, bindingInfos); } return bindingInfos; } return null; } /** * Adds a {@link BindingInfo} object to the internal list associated with the specified module. * <p> * This method returns silently, if any of the parameters is {@code null}. * * @param bundle the module to which the binding information to be added * @param bindingInfo the binding information to be added */ public synchronized void addBindingInfo(Bundle bundle, BindingInfo bindingInfo) { if (bindingInfo != null) { List<BindingInfo> bindingInfos = acquireBindingInfos(bundle); if (bindingInfos != null) { bindingInfos.add(bindingInfo); } } } /** * Removes all {@link BindingInfo} objects from the internal list * associated with the specified module. * <p> * This method returns silently if the module is {@code null}. * * @param bundle the module for which all associated binding informations to be removed */ public synchronized void removeAllBindingInfos(Bundle bundle) { if (bundle != null) { List<BindingInfo> bindingInfos = this.bundleBindingInfoMap.get(bundle); if (bindingInfos != null) { this.bundleBindingInfoMap.remove(bundle); } } } @Override public synchronized BindingInfo getBindingInfo(String id, Locale locale) { Collection<Entry<Bundle, List<BindingInfo>>> bindingInfoList = this.bundleBindingInfoMap.entrySet(); if (bindingInfoList != null) { for (Entry<Bundle, List<BindingInfo>> bindingInfos : bindingInfoList) { for (BindingInfo bindingInfo : bindingInfos.getValue()) { if (bindingInfo.getId().equals(id)) { return createLocalizedBindingInfo(bindingInfos.getKey(), bindingInfo, locale); } } } } return null; } @Override public synchronized Set<BindingInfo> getBindingInfos(Locale locale) { Set<BindingInfo> allBindingInfos = new LinkedHashSet<>(10); Collection<Entry<Bundle, List<BindingInfo>>> bindingInfoSet = this.bundleBindingInfoMap.entrySet(); if (bindingInfoSet != null) { for (Entry<Bundle, List<BindingInfo>> bindingInfos : bindingInfoSet) { for (BindingInfo bindingInfo : bindingInfos.getValue()) { BindingInfo localizedBindingInfo = createLocalizedBindingInfo(bindingInfos.getKey(), bindingInfo, locale); allBindingInfos.add(localizedBindingInfo); } } } return allBindingInfos; } @Reference public void setI18nProvider(I18nProvider i18nProvider) { this.bindingI18nUtil = new BindingI18nUtil(i18nProvider); } public void unsetI18nProvider(I18nProvider i18nProvider) { this.bindingI18nUtil = null; } @Reference(target = "(esh.scope=core.xml.binding)") public void setConfigDescriptionProvider(ConfigDescriptionProvider configDescriptionProvider) { this.configDescriptionProvider = (AbstractXmlConfigDescriptionProvider) configDescriptionProvider; } public void unsetConfigDescriptionProvider(ConfigDescriptionProvider configDescriptionProvider) { this.configDescriptionProvider = null; } private BindingInfo createLocalizedBindingInfo(Bundle bundle, BindingInfo bindingInfo, Locale locale) { if (this.bindingI18nUtil != null) { String name = this.bindingI18nUtil.getName(bundle, bindingInfo.getId(), bindingInfo.getName(), locale); String description = this.bindingI18nUtil.getDescription(bundle, bindingInfo.getId(), bindingInfo.getDescription(), locale); return new BindingInfo(bindingInfo.getId(), name, description, bindingInfo.getAuthor(), bindingInfo.getServiceId(), bindingInfo.getConfigDescriptionURI()); } else { return bindingInfo; } } }