2012-03-14 49 views
3

我正在運行JBoss AS 7.1.0.CR1b。我在standalone.xml中定義了幾個數據源,例如如何在Java代碼中列出JBoss AS 7數據源屬性?

 <subsystem xmlns="urn:jboss:domain:datasources:1.0"> 
      <datasources> 
       <datasource jndi-name="java:/MyDS" pool-name="MyDS_Pool" enabled="true" use-java-context="true" use-ccm="true"> 
        <connection-url>some-url</connection-url> 
        <driver>the-driver</driver> 
        [etc] 

一切工作正常。

我試圖訪問我的代碼中包含的信息 - 特別是connection-urldriver屬性。

我已經試過正從JNDI數據源,該是正常的,但它似乎沒有提供訪問這些屬性:從這個數據源

// catches removed 
InitialContext context; 
DataSource dataSource = null; 
context = new InitialContext(); 
dataSource = (DataSource) context.lookup(jndi); 

ClientInfo客戶端和DatabaseMetaData中從一個Connection對象也不要」 t也包含這些粒度的JBoss屬性。

我的代碼將在容器內運行,並且數據源已被指定,所以所有代碼都應該可用。我查看了IronJacamar接口org.jboss.jca.common.api.metadata.ds.DataSource及其實現類,這些接口似乎對我所需的信息具有可訪問的鉤子,但我找不到有關如何使用容器中已部署的資源創建此類對象的任何信息(在impl上只有構造函數需要手動輸入所有屬性)。

JBoss AS 7的命令行界面允許您導航數據源並將其列爲目錄系統。 http://www.paykin.info/java/add-datasource-programaticaly-cli-jboss-7/提供了一篇關於如何使用我認爲是Java Management API與子系統進行交互的優秀文章,但這似乎涉及連接到目標JBoss服務器。我的代碼已經在運行那個服務器,所以肯定有一個更簡單的方法來做到這一點?

希望有人能幫助。非常感謝。

回答

5

你真正想要做的是管理行動。最好的方法是使用可用的管理API。

下面是一個簡單的獨立例如:

public class Main { 

    public static void main(final String[] args) throws Exception { 
     final List<ModelNode> dataSources = getDataSources(); 
     for (ModelNode dataSource : dataSources) { 
      System.out.printf("Datasource: %s%n", dataSource.asString()); 
     } 
    } 

    public static List<ModelNode> getDataSources() throws IOException { 
     final ModelNode request = new ModelNode(); 
     request.get(ClientConstants.OP).set("read-resource"); 
     request.get("recursive").set(true); 
     request.get(ClientConstants.OP_ADDR).add("subsystem", "datasources"); 
     ModelControllerClient client = null; 
     try { 
      client = ModelControllerClient.Factory.create(InetAddress.getByName("127.0.0.1"), 9999); 
      final ModelNode response = client.execute(new OperationBuilder(request).build()); 
      reportFailure(response); 
      return response.get(ClientConstants.RESULT).get("data-source").asList(); 
     } finally { 
      safeClose(client); 
     } 
    } 

    public static void safeClose(final Closeable closeable) { 
     if (closeable != null) try { 
      closeable.close(); 
     } catch (Exception e) { 
      // no-op 
     } 
    } 


    private static void reportFailure(final ModelNode node) { 
     if (!node.get(ClientConstants.OUTCOME).asString().equals(ClientConstants.SUCCESS)) { 
      final String msg; 
      if (node.hasDefined(ClientConstants.FAILURE_DESCRIPTION)) { 
       if (node.hasDefined(ClientConstants.OP)) { 
        msg = String.format("Operation '%s' at address '%s' failed: %s", node.get(ClientConstants.OP), node.get(ClientConstants.OP_ADDR), node.get(ClientConstants.FAILURE_DESCRIPTION)); 
       } else { 
        msg = String.format("Operation failed: %s", node.get(ClientConstants.FAILURE_DESCRIPTION)); 
       } 
      } else { 
       msg = String.format("Operation failed: %s", node); 
      } 
      throw new RuntimeException(msg); 
     } 
    } 
} 

我能想到的唯一的另一種方法是添加模塊依賴於服務器內部。它可以完成,但我可能會首先使用管理API。

+0

非常感謝@ JamesR.Perkins,看起來不錯 - 感謝您花時間寫下來。我們必須使用'InetAddress.getByName(「127.0.0.1」),9999)創建客戶端這一事實表明,它涉及某種外部調用服務器,但如果你明白我的意思,當我們已經*在該環境中運行*。希望我可以注入一些JBoss類的實例來獲取子系統/數據源屬性,而不需要明確的外部調用。如果不是,我肯定會使用你的建議。再次感謝。 – 2012-03-15 10:35:35

+0

@BenKirby AFAIK沒有什麼可以注入的。這實際上是一個管理操作,這就是爲什麼它需要客戶端和遠程樣式連接。如果我發現其他可以工作的東西或想出其他東西,我會編輯答案。 :-) – 2012-03-15 15:55:36

+0

找到JBoss CLI公共API [https://community.jboss.org/wiki/JBossAS7Command-linePublicAPI](https://community.jboss.org/wiki/JBossAS7Command-linePublicAPI),其中_appears_不需要遠程連接,但我目前無法升級到他們推薦的JBoss版本(「自JBoss AS 7.1.1以來」)。所以已經與您的解決方案 - 再次感謝您的答覆。 – 2012-03-26 15:17:52