如何将 bundle configuration/property 文件放入 /etc karaf 文件夹

How to place bundle configuration/property file in /etc karaf folder

我想部署karaf中部署的bundle的所有配置文件放在karaf etc文件夹中。我希望当束 conf 文件更改时,karaf 会通知。

我有一个包含几个特性的发行版 XML.I 的一个特性的例子已经尝试了几个东西,例如我将 conf 文件添加到功能中,如下所示,但这不起作用。

<feature name="gc-backbone-mqtt" version="${linksmart.gc.version}">
    <feature version="${linksmart.gc.version}">gc-backbone-router</feature>
    <bundle>mvn:org.eclipse.paho/org.eclipse.paho.client.mqttv3/1.0.0</bundle>
    <feature version="${linksmart.gc.version}">gc-type-tunnelled</feature>
    <configfile finalname="/etc/mqttBackboneProtocol.cfg">mvn:eu.linksmart.gc/backbone.mqtt.impl/${linksmart.gc.version}/mqttprotocol.properties</configfile>
    <bundle>mvn:eu.linksmart.gc/backbone.mqtt.impl/${linksmart.gc.version}</bundle>
</feature>

我尝试过的一些事情:

http://karaf.922171.n3.nabble.com/OSGi-bundle-configuration-file-td4025438.html

http://www.liquid-reality.de/display/liquid/2011/09/23/Karaf+Tutorial+Part+2+-+Using+the+Configuration+Admin+Service

我不想复制具有特定路径的文件,如下所示:

有人知道怎么做吗?

更新

为了实现将配置文件部署到 etc 文件夹中以便可以在外部重新配置捆绑包,我分 3 个步骤完成了:

构建配置文件:(Works)

为了使 Maven 可以寻址配置文件,我在 bundle pom.xml 中添加了以下部分。通过这种方式,配置文件部署在存储库中:

pom.xml

 <build>
        <plugins>
            <plugin>
                <groupId>org.codehaus.mojo</groupId>
                <artifactId>build-helper-maven-plugin</artifactId>
                <executions>
                    <execution>
                        <id>attach-artifacts</id>
                        <phase>package</phase>
                        <goals>
                            <goal>attach-artifact</goal>
                        </goals>
                        <configuration>
                            <artifacts>
                                <artifact>
                                    <file>src/main/resources/mqttprotocol.properties</file>
                                    <type>cfg</type>
                                    <classifier>configuration</classifier>
                                </artifact>
                            </artifacts>
                        </configuration>
                    </execution>
                </executions>
            </plugin>

        </plugins>
    </build>

在 karaf 中部署文件 etc (Works)

为了在 karaf etc 文件夹中部署配置文件,我在功能文件中添加了 <configfile>,如下所示:

features.xml

    <feature name="gc-backbone-mqtt" version="${linksmart.gc.version}">
        <feature version="${linksmart.gc.version}">gc-backbone-router</feature>
        <bundle>mvn:org.eclipse.paho/org.eclipse.paho.client.mqttv3/1.0.0</bundle>
        <bundle>mvn:org.apache.felix/org.apache.felix.fileinstall/3.2.8</bundle>
        <configfile finalname="/etc/MQTTBackboneProtocol.cfg">mvn:eu.linksmart.gc/network.backbone.protocol.mqtt.impl/${linksmart.gc.version}/cfg/configuration</configfile>
        <feature version="${linksmart.gc.version}">gc-type-tunnelled</feature>          <bundle>mvn:eu.linksmart.gc/network.backbone.protocol.mqtt.impl/${linksmart.gc.version}</bundle>
    </feature>

捕获配置更改:(不工作

为了捕获配置文件的更改,我添加了您建议的代码 (@Donald_W)。问题是我只收到文件通知,文件位于文件夹 deploy 而不是 etc。我调试了这段代码,发现 etc 中的文件被专门称为这些文件的 "listeners"。我不知道如何才能成为部署在 etc

中的文件的侦听器

您可以使用 Karaf 内置的 Felix 文件安装器,它会在 etc 下的文件更改时给您回调。

一旦您将实现 ArtifactInstaller 的服务发布到 OSGi 服务注册表中,它就会被 FileInstaller 检测到 - 即 Whiteboard pattern.

<dependency>
    <groupId>org.apache.felix</groupId>
    <artifactId>org.apache.felix.fileinstall</artifactId>
    <version>3.4.2</version>
    <scope>provided</scope>
</dependency>

示例代码(使用 iPojo,但 Blueprint/DS 也能正常工作)为:

package com.example.deployer.internal;

import org.apache.felix.fileinstall.ArtifactInstaller;
import org.apache.felix.ipojo.annotations.Component;
import org.apache.felix.ipojo.annotations.Instantiate;
import org.apache.felix.ipojo.annotations.Provides;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;


@Instantiate()
@Component(name = "propsDeployer")
@Provides(specifications = ArtifactInstaller.class)
public class PropsDeployer implements ArtifactInstaller {
    Logger logger = LoggerFactory.getLogger(JsonDeployer.class);

    @Override
    public void install(File artifact) throws Exception {
        logOutput("Installing",artifact);
    }

    @Override
    public void update(File artifact) throws Exception {
        logOutput("Updating",artifact);
    }

    @Override
    public void uninstall(File artifact) throws Exception {
        logger.info("Uninstalling artifact: {}", artifact.getName());
    }

    @Override
    public boolean canHandle(File artifact) {
        return artifact.getName().endsWith(".props");
    }

    private void logOutput(String action, File artifact) throws IOException {
        logger.info(action + " artifact: {}", artifact.getName());
        Properties props = new Properties();
        FileReader in = null;
        try {
            in = new FileReader(artifact.getCanonicalFile());
            props.load(in);
        } finally {
            if (in != null) {
                in.close();
            }
        }

        // Do whatever you want here:

        for(Object key: props.keySet()) {
            logger.info(action + " Property received: {} {}",key,props.get(key));
        }
    }
}

应该给你这样的输出:

2015-04-27 20:16:53,726 | INFO  | ime/karaf/deploy | PropsDeployer                     | 101 - com.example.scratch.deployer - 1.0.0.SNAPSHOT | Updating artifact: my.json
2015-04-27 20:16:53,728 | INFO  | ime/karaf/deploy | PropsDeployer                     | 101 - com.example.scratch.deployer - 1.0.0.SNAPSHOT | Updating Property received: myprop myval
2015-04-27 20:16:53,728 | INFO  | ime/karaf/deploy | PropsDeployer                     | 101 - com.example.scratch.deployer - 1.0.0.SNAPSHOT | Updating Property received: hello world

您需要一个唯一的文件扩展名。 .cfg 将被您说不想使用的 ConfigurationAdmin 服务捕获。

.props(如上)之类的东西就可以了。

顺便说一句,你真的应该考虑使用ConfigurationAdmin。它非常强大。 karaf 中内置了用于管理配置的命令,您所做的任何更改都将保存回 .cfg 文件。

尤里卡!

正如我在更新中提到的,第 1 步(使用 Maven 构建配置文件)和第 2 步(部署 conf 文件 etc)正在运行,但捆绑包和配置文件未连接。显然,原因是配置文件是用另一个 PID 作为注册到它的服务安装的。为了解决这个问题,我将 ManageService 注册到部署的配置文件的 PID。在我的例子中看起来如下:

 Hashtable <String, Object> properties = new Hashtable<String, Object>();
 properties.put(Constants.SERVICE_PID, "MQTTBackboneProtocol");
 context.registerService (ManagedService.class.getName(),this , properties);

其中 this 是实现 ManageService 的 class 并且 PID 必须与 etc 中部署的配置相同,在这种情况下 "MQTTBackboneProtocol" 由于功能定义指示配置文件为:

<configfile finalname="/etc/MQTTBackboneProtocol.cfg">mvn:eu.linksmart.gc/network.backbone.protocol.mqtt.impl/${linksmart.gc.version}/cfg/configuration</configfile>