Add a YAML file provider for semantic tags (#3659)

* Add a YAML file provider for semantic tags

Files in folder conf/tags are loaded by this provider.

Related to #3619

Signed-off-by: Laurent Garnier <lg.hc@free.fr>
pull/3910/head
lolodomo 2023-12-10 11:26:03 +01:00 committed by GitHub
parent 152ffe3fbb
commit 070de55b27
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
17 changed files with 835 additions and 0 deletions

View File

@ -514,6 +514,12 @@
<version>${project.version}</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>org.openhab.core.bundles</groupId>
<artifactId>org.openhab.core.model.yaml</artifactId>
<version>${project.version}</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>org.openhab.core.bundles</groupId>
<artifactId>org.openhab.core.ui</artifactId>

View File

@ -0,0 +1,29 @@
<?xml version="1.0" encoding="UTF-8"?>
<classpath>
<classpathentry kind="src" output="target/classes" path="src/main/java">
<attributes>
<attribute name="optional" value="true"/>
<attribute name="maven.pomderived" value="true"/>
</attributes>
</classpathentry>
<classpathentry kind="con" path="org.eclipse.jdt.launching.JRE_CONTAINER/org.eclipse.jdt.internal.debug.ui.launcher.StandardVMType/JavaSE-17">
<attributes>
<attribute name="maven.pomderived" value="true"/>
<attribute name="annotationpath" value="target/dependency"/>
</attributes>
</classpathentry>
<classpathentry kind="con" path="org.eclipse.m2e.MAVEN2_CLASSPATH_CONTAINER">
<attributes>
<attribute name="maven.pomderived" value="true"/>
<attribute name="annotationpath" value="target/dependency"/>
</attributes>
</classpathentry>
<classpathentry kind="src" output="target/test-classes" path="src/test/java">
<attributes>
<attribute name="optional" value="true"/>
<attribute name="maven.pomderived" value="true"/>
<attribute name="test" value="true"/>
</attributes>
</classpathentry>
<classpathentry kind="output" path="target/classes"/>
</classpath>

View File

@ -0,0 +1,23 @@
<?xml version="1.0" encoding="UTF-8"?>
<projectDescription>
<name>org.openhab.core.model.yaml</name>
<comment></comment>
<projects>
</projects>
<buildSpec>
<buildCommand>
<name>org.eclipse.jdt.core.javabuilder</name>
<arguments>
</arguments>
</buildCommand>
<buildCommand>
<name>org.eclipse.m2e.core.maven2Builder</name>
<arguments>
</arguments>
</buildCommand>
</buildSpec>
<natures>
<nature>org.eclipse.jdt.core.javanature</nature>
<nature>org.eclipse.m2e.core.maven2Nature</nature>
</natures>
</projectDescription>

View File

@ -0,0 +1,14 @@
This content is produced and maintained by the openHAB project.
* Project home: https://www.openhab.org
== Declared Project Licenses
This program and the accompanying materials are made available under the terms
of the Eclipse Public License 2.0 which is available at
https://www.eclipse.org/legal/epl-2.0/.
== Source Code
https://github.com/openhab/openhab-core

View File

@ -0,0 +1,29 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.openhab.core.bundles</groupId>
<artifactId>org.openhab.core.reactor.bundles</artifactId>
<version>4.1.0-SNAPSHOT</version>
</parent>
<artifactId>org.openhab.core.model.yaml</artifactId>
<name>openHAB Core :: Bundles :: Model YAML</name>
<dependencies>
<dependency>
<groupId>org.openhab.core.bundles</groupId>
<artifactId>org.openhab.core</artifactId>
<version>${project.version}</version>
</dependency>
<dependency>
<groupId>org.openhab.core.bundles</groupId>
<artifactId>org.openhab.core.semantics</artifactId>
<version>${project.version}</version>
</dependency>
</dependencies>
</project>

View File

@ -0,0 +1,65 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml;
import java.util.List;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link AbstractYamlFile} is the DTO base class used to map a YAML configuration file.
*
* A YAML configuration file consists of a version and a list of elements.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
public abstract class AbstractYamlFile implements YamlFile {
private final Logger logger = LoggerFactory.getLogger(AbstractYamlFile.class);
/**
* YAML file version
*/
public int version;
@Override
public abstract List<? extends YamlElement> getElements();
@Override
public int getVersion() {
return version;
}
@Override
public boolean isValid() {
// Checking duplicated elements
List<? extends YamlElement> elts = getElements();
long nbDistinctIds = elts.stream().map(YamlElement::getId).distinct().count();
if (nbDistinctIds < elts.size()) {
logger.debug("Elements with same ids detected in the file");
return false;
}
// Checking each element
for (int i = 0; i < elts.size(); i++) {
if (!elts.get(i).isValid()) {
logger.debug("Error in element {}", i + 1);
return false;
}
}
return true;
}
}

View File

@ -0,0 +1,39 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link YamlElement} interface offers an identifier and a check validity method
* to any element defined in a YAML configuration file.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
public interface YamlElement {
/**
* Get the identifier of the YAML element
*
* @return the identifier as a string
*/
String getId();
/**
* Check that the YAML element is valid
*
* @return true if all the checks are OK
*/
boolean isValid();
}

View File

@ -0,0 +1,48 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml;
import java.util.List;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link YamlFile} is the interface to manage the generic content of a YAML configuration file.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
public interface YamlFile {
/**
* Get the list of elements present in the YAML file.
*
* @return the list of elements
*/
List<? extends YamlElement> getElements();
/**
* Get the version present in the YAML file.
*
* @return the version in the file
*/
int getVersion();
/**
* Check that the file content is valid.
* It includes the check of duplicated elements (same identifier) and the check of each element.
*
* @return true if all the checks are OK
*/
boolean isValid();
}

View File

@ -0,0 +1,75 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml;
import java.util.Collection;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link YamlModelListener} interface is responsible for managing a particular model type
* with data processed from YAML configuration files.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
public interface YamlModelListener<T extends YamlElement> {
/**
* Method called by the model repository when elements from a model are added.
*
* @param modelName the name of the model
* @param elements the collection of added elements
*/
void addedModel(String modelName, Collection<? extends YamlElement> elements);
/**
* Method called by the model repository when elements from a model are updated.
*
* @param modelName the name of the model
* @param elements the collection of updated elements
*/
void updatedModel(String modelName, Collection<? extends YamlElement> elements);
/**
* Method called by the model repository when elements from a model are removed.
*
* @param modelName the name of the model
* @param elements the collection of removed elements
*/
void removedModel(String modelName, Collection<? extends YamlElement> elements);
/**
* Get the root name of this model type which is also the name of the root folder
* containing the user files for this model type.
*
* A path is unexpected. What is expected is for example "items" or "things".
*
* @return the model root name
*/
String getRootName();
/**
* Get the DTO class to be used for a file providing objects for this model type.
*
* @return the DTO file class
*/
Class<? extends AbstractYamlFile> getFileClass();
/**
* Get the DTO class to be used for each object of this model type.
*
* @return the DTO element class
*/
Class<T> getElementClass();
}

View File

@ -0,0 +1,39 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link YamlParseException} is used when an error is detected when parsing the content
* of a YAML configuration file.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
public class YamlParseException extends Exception {
private static final long serialVersionUID = 1L;
public YamlParseException(String message) {
super(message);
}
public YamlParseException(Throwable cause) {
super(cause);
}
public YamlParseException(String message, Throwable cause) {
super(message, cause);
}
}

View File

@ -0,0 +1,198 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml.internal;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.model.yaml.AbstractYamlFile;
import org.openhab.core.model.yaml.YamlElement;
import org.openhab.core.model.yaml.YamlModelListener;
import org.openhab.core.model.yaml.YamlParseException;
import org.openhab.core.service.WatchService;
import org.openhab.core.service.WatchService.Kind;
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;
import org.osgi.service.component.annotations.ReferenceCardinality;
import org.osgi.service.component.annotations.ReferencePolicy;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.dataformat.yaml.YAMLFactory;
/**
* The {@link YamlModelRepository} is an OSGi service, that encapsulates all YAML file processing
* including file monitoring to detect created, updated and removed YAML configuration files.
* Data processed from these files are consumed by registered OSGi services that implement {@link YamlModelListener}.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
@Component(immediate = true)
public class YamlModelRepository implements WatchService.WatchEventListener {
private final Logger logger = LoggerFactory.getLogger(YamlModelRepository.class);
private final WatchService watchService;
private final Path watchPath;
private final ObjectMapper yamlReader;
private final Map<String, List<YamlModelListener<?>>> listeners = new ConcurrentHashMap<>();
private final Map<Path, List<? extends YamlElement>> objects = new ConcurrentHashMap<>();
@Activate
public YamlModelRepository(@Reference(target = WatchService.CONFIG_WATCHER_FILTER) WatchService watchService) {
this.watchService = watchService;
this.yamlReader = new ObjectMapper(new YAMLFactory());
yamlReader.findAndRegisterModules();
watchService.registerListener(this, Path.of(""));
watchPath = watchService.getWatchPath();
}
@Deactivate
public void deactivate() {
watchService.unregisterListener(this);
}
// The method is "synchronized" to avoid concurrent files processing
@Override
public synchronized void processWatchEvent(Kind kind, Path path) {
Path fullPath = watchPath.resolve(path);
String dirName = path.subpath(0, 1).toString();
if (Files.isDirectory(fullPath) || fullPath.toFile().isHidden() || !fullPath.toString().endsWith(".yaml")) {
logger.trace("Ignored {}", fullPath);
return;
}
getListeners(dirName).forEach(listener -> processWatchEvent(dirName, kind, fullPath, listener));
}
private void processWatchEvent(String dirName, Kind kind, Path fullPath, YamlModelListener<?> listener) {
logger.debug("processWatchEvent dirName={} kind={} fullPath={} listener={}", dirName, kind, fullPath,
listener.getClass().getSimpleName());
Map<String, ? extends YamlElement> oldObjects;
Map<String, ? extends YamlElement> newObjects;
if (kind == WatchService.Kind.DELETE) {
newObjects = Map.of();
List<? extends YamlElement> oldListObjects = objects.remove(fullPath);
if (oldListObjects == null) {
oldListObjects = List.of();
}
oldObjects = oldListObjects.stream().collect(Collectors.toMap(YamlElement::getId, obj -> obj));
} else {
AbstractYamlFile yamlData;
try {
yamlData = readYamlFile(fullPath, listener.getFileClass());
} catch (YamlParseException e) {
logger.warn("Failed to parse Yaml file {} with DTO class {}: {}", fullPath,
listener.getFileClass().getName(), e.getMessage());
return;
}
List<? extends YamlElement> newListObjects = yamlData.getElements();
newObjects = newListObjects.stream().collect(Collectors.toMap(YamlElement::getId, obj -> obj));
List<? extends YamlElement> oldListObjects = objects.get(fullPath);
if (oldListObjects == null) {
oldListObjects = List.of();
}
oldObjects = oldListObjects.stream().collect(Collectors.toMap(YamlElement::getId, obj -> obj));
objects.put(fullPath, newListObjects);
}
String modelName = fullPath.toFile().getName();
modelName = modelName.substring(0, modelName.indexOf(".yaml"));
List<? extends YamlElement> listElements;
listElements = oldObjects.entrySet().stream()
.filter(entry -> entry.getValue().getClass().equals(listener.getElementClass())
&& !newObjects.containsKey(entry.getKey()))
.map(Map.Entry::getValue).toList();
if (!listElements.isEmpty()) {
listener.removedModel(modelName, listElements);
}
listElements = newObjects.entrySet().stream()
.filter(entry -> entry.getValue().getClass().equals(listener.getElementClass())
&& !oldObjects.containsKey(entry.getKey()))
.map(Map.Entry::getValue).toList();
if (!listElements.isEmpty()) {
listener.addedModel(modelName, listElements);
}
// Object is ignored if unchanged
listElements = newObjects.entrySet().stream()
.filter(entry -> entry.getValue().getClass().equals(listener.getElementClass())
&& oldObjects.containsKey(entry.getKey())
&& !entry.getValue().equals(oldObjects.get(entry.getKey())))
.map(Map.Entry::getValue).toList();
if (!listElements.isEmpty()) {
listener.updatedModel(modelName, listElements);
}
}
private AbstractYamlFile readYamlFile(Path path, Class<? extends AbstractYamlFile> dtoClass)
throws YamlParseException {
logger.debug("readYamlFile {} with {}", path.toFile().getAbsolutePath(), dtoClass.getName());
try {
AbstractYamlFile dto = yamlReader.readValue(path.toFile(), dtoClass);
if (!dto.isValid()) {
throw new YamlParseException("The file is not valid, some checks failed!");
}
return dto;
} catch (IOException e) {
throw new YamlParseException(e);
}
}
@Reference(cardinality = ReferenceCardinality.MULTIPLE, policy = ReferencePolicy.DYNAMIC)
protected void addYamlModelListener(YamlModelListener<?> listener) {
String dirName = listener.getRootName();
logger.debug("Adding model listener for {}", dirName);
getListeners(dirName).add(listener);
// Load all existing YAML files
try (Stream<Path> stream = Files.walk(watchPath.resolve(dirName))) {
stream.forEach(path -> {
if (!Files.isDirectory(path) && !path.toFile().isHidden() && path.toString().endsWith(".yaml")) {
processWatchEvent(dirName, Kind.CREATE, path, listener);
}
});
} catch (IOException ignored) {
}
}
protected void removeYamlModelListener(YamlModelListener<?> listener) {
String dirName = listener.getRootName();
logger.debug("Removing model listener for {}", dirName);
getListeners(dirName).remove(listener);
}
private List<YamlModelListener<?>> getListeners(String dirName) {
return Objects.requireNonNull(listeners.computeIfAbsent(dirName, k -> new CopyOnWriteArrayList<>()));
}
}

View File

@ -0,0 +1,66 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml.internal.semantics;
import java.util.List;
import java.util.Objects;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.core.model.yaml.YamlElement;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link YamlSemanticTag} is a data transfer object used to serialize a semantic tag
* in a YAML configuration file.
*
* @author Laurent Garnier - Initial contribution
*/
public class YamlSemanticTag implements YamlElement {
private final Logger logger = LoggerFactory.getLogger(YamlSemanticTag.class);
public String uid;
public String label;
public String description;
public List<String> synonyms;
public YamlSemanticTag() {
}
@Override
public String getId() {
return uid;
}
@Override
public boolean isValid() {
if (uid == null) {
logger.debug("uid missing");
return false;
}
return true;
}
@Override
public boolean equals(@Nullable Object obj) {
if (this == obj) {
return true;
} else if (obj == null || getClass() != obj.getClass()) {
return false;
}
YamlSemanticTag that = (YamlSemanticTag) obj;
return Objects.equals(uid, that.uid) && Objects.equals(label, that.label)
&& Objects.equals(description, that.description) && Objects.equals(synonyms, that.synonyms);
}
}

View File

@ -0,0 +1,126 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml.internal.semantics;
import java.util.Collection;
import java.util.Comparator;
import java.util.List;
import java.util.Set;
import java.util.TreeSet;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.common.registry.AbstractProvider;
import org.openhab.core.model.yaml.AbstractYamlFile;
import org.openhab.core.model.yaml.YamlElement;
import org.openhab.core.model.yaml.YamlModelListener;
import org.openhab.core.semantics.SemanticTag;
import org.openhab.core.semantics.SemanticTagImpl;
import org.openhab.core.semantics.SemanticTagProvider;
import org.openhab.core.semantics.SemanticTagRegistry;
import org.osgi.service.component.annotations.Activate;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Deactivate;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* {@link YamlSemanticTagProvider} is an OSGi service, that allows to define semantic tags
* in YAML configuration files in folder conf/tags.
* Files can be added, updated or removed at runtime.
* These semantic tags are automatically exposed to the {@link SemanticTagRegistry}.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
@Component(immediate = true, service = { SemanticTagProvider.class, YamlSemanticTagProvider.class,
YamlModelListener.class })
public class YamlSemanticTagProvider extends AbstractProvider<SemanticTag>
implements SemanticTagProvider, YamlModelListener<YamlSemanticTag> {
private final Logger logger = LoggerFactory.getLogger(YamlSemanticTagProvider.class);
private final Set<SemanticTag> tags = new TreeSet<>(Comparator.comparing(SemanticTag::getUID));
@Activate
public YamlSemanticTagProvider() {
}
@Deactivate
public void deactivate() {
tags.clear();
}
@Override
public Collection<SemanticTag> getAll() {
return tags;
}
@Override
public String getRootName() {
return "tags";
}
@Override
public Class<? extends AbstractYamlFile> getFileClass() {
return YamlSemanticTags.class;
}
@Override
public Class<YamlSemanticTag> getElementClass() {
return YamlSemanticTag.class;
}
@Override
public void addedModel(String modelName, Collection<? extends YamlElement> elements) {
List<SemanticTag> added = elements.stream().map(e -> mapSemanticTag((YamlSemanticTag) e))
.sorted(Comparator.comparing(SemanticTag::getUID)).toList();
tags.addAll(added);
added.forEach(t -> {
logger.debug("model {} added tag {}", modelName, t.getUID());
notifyListenersAboutAddedElement(t);
});
}
@Override
public void updatedModel(String modelName, Collection<? extends YamlElement> elements) {
List<SemanticTag> updated = elements.stream().map(e -> mapSemanticTag((YamlSemanticTag) e)).toList();
updated.forEach(t -> {
tags.stream().filter(tag -> tag.getUID().equals(t.getUID())).findFirst().ifPresentOrElse(oldTag -> {
tags.remove(oldTag);
tags.add(t);
logger.debug("model {} updated tag {}", modelName, t.getUID());
notifyListenersAboutUpdatedElement(oldTag, t);
}, () -> logger.debug("model {} tag {} not found", modelName, t.getUID()));
});
}
@Override
public void removedModel(String modelName, Collection<? extends YamlElement> elements) {
List<SemanticTag> removed = elements.stream().map(e -> mapSemanticTag((YamlSemanticTag) e))
.sorted(Comparator.comparing(SemanticTag::getUID).reversed()).toList();
removed.forEach(t -> {
tags.stream().filter(tag -> tag.getUID().equals(t.getUID())).findFirst().ifPresentOrElse(oldTag -> {
tags.remove(oldTag);
logger.debug("model {} removed tag {}", modelName, t.getUID());
notifyListenersAboutRemovedElement(oldTag);
}, () -> logger.debug("model {} tag {} not found", modelName, t.getUID()));
});
}
private SemanticTag mapSemanticTag(YamlSemanticTag tagDTO) {
if (tagDTO.uid == null) {
throw new IllegalArgumentException("The argument 'tagDTO.uid' must not be null.");
}
return new SemanticTagImpl(tagDTO.uid, tagDTO.label, tagDTO.description, tagDTO.synonyms);
}
}

View File

@ -0,0 +1,36 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.core.model.yaml.internal.semantics;
import java.util.List;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.model.yaml.AbstractYamlFile;
import org.openhab.core.model.yaml.YamlElement;
/**
* The {@link YamlSemanticTags} is a data transfer object used to serialize a list of semantic tags
* in a YAML configuration file.
*
* @author Laurent Garnier - Initial contribution
*/
@NonNullByDefault
public class YamlSemanticTags extends AbstractYamlFile {
public List<YamlSemanticTag> tags = List.of();
@Override
public List<? extends YamlElement> getElements() {
return tags;
}
}

View File

@ -22,6 +22,7 @@ import java.util.TreeMap;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.common.registry.AbstractProvider;
import org.openhab.core.common.registry.RegistryChangeListener;
import org.openhab.core.items.GroupItem;
import org.openhab.core.items.Item;
import org.openhab.core.items.ItemRegistry;
@ -33,6 +34,7 @@ import org.openhab.core.semantics.Equipment;
import org.openhab.core.semantics.Location;
import org.openhab.core.semantics.Point;
import org.openhab.core.semantics.Property;
import org.openhab.core.semantics.SemanticTag;
import org.openhab.core.semantics.SemanticTagRegistry;
import org.openhab.core.semantics.SemanticTags;
import org.openhab.core.semantics.Tag;
@ -72,11 +74,16 @@ public class SemanticsMetadataProvider extends AbstractProvider<Metadata>
private final Map<String, Metadata> semantics = new TreeMap<>(String::compareTo);
private final ItemRegistry itemRegistry;
private final SemanticTagRegistry semanticTagRegistry;
private SemanticTagRegistryChangeListener listener;
@Activate
public SemanticsMetadataProvider(final @Reference ItemRegistry itemRegistry,
final @Reference SemanticTagRegistry semanticTagRegistry) {
this.itemRegistry = itemRegistry;
this.semanticTagRegistry = semanticTagRegistry;
this.listener = new SemanticTagRegistryChangeListener(this);
}
@Activate
@ -86,10 +93,12 @@ public class SemanticsMetadataProvider extends AbstractProvider<Metadata>
processItem(item);
}
itemRegistry.addRegistryChangeListener(this);
semanticTagRegistry.addRegistryChangeListener(listener);
}
@Deactivate
protected void deactivate() {
semanticTagRegistry.removeRegistryChangeListener(listener);
itemRegistry.removeRegistryChangeListener(this);
semantics.clear();
}
@ -280,4 +289,28 @@ public class SemanticsMetadataProvider extends AbstractProvider<Metadata>
public void updated(Item oldItem, Item item) {
processItem(item);
}
private class SemanticTagRegistryChangeListener implements RegistryChangeListener<SemanticTag> {
private SemanticsMetadataProvider provider;
public SemanticTagRegistryChangeListener(SemanticsMetadataProvider provider) {
this.provider = provider;
}
@Override
public void added(SemanticTag element) {
provider.allItemsChanged(List.of());
}
@Override
public void removed(SemanticTag element) {
provider.allItemsChanged(List.of());
}
@Override
public void updated(SemanticTag oldElement, SemanticTag element) {
provider.allItemsChanged(List.of());
}
}
}

View File

@ -102,6 +102,7 @@
<module>org.openhab.core.model.thing</module>
<module>org.openhab.core.model.thing.ide</module>
<module>org.openhab.core.model.thing.runtime</module>
<module>org.openhab.core.model.yaml</module>
<module>org.openhab.core.storage.json</module>
<module>org.openhab.core.test</module>
<module>org.openhab.core.test.magic</module>

View File

@ -390,6 +390,13 @@
<bundle>mvn:org.openhab.core.bundles/org.openhab.core.model.lsp/${project.version}</bundle>
</feature>
<feature name="openhab-core-model-yaml" version="${project.version}">
<feature>openhab-core-base</feature>
<bundle>mvn:org.openhab.core.bundles/org.openhab.core.model.yaml/${project.version}</bundle>
<requirement>openhab.tp;filter:="(feature=jackson)"</requirement>
<feature dependency="true">openhab.tp-jackson</feature>
</feature>
<feature name="openhab-core-storage-json" version="${project.version}">
<feature>openhab-core-base</feature>
@ -434,6 +441,7 @@
<feature>openhab-core-model-script</feature>
<feature>openhab-core-model-sitemap</feature>
<feature>openhab-core-model-thing</feature>
<feature>openhab-core-model-yaml</feature>
<feature>openhab-core-ui-icon</feature>
<feature>openhab-core-storage-json</feature>
<feature>openhab-runtime-certificate</feature>