java調(diào)用kettle文件,根據(jù)環(huán)境修改數(shù)據(jù)源

在使用kettle可視化工具進(jìn)行編輯生成kettle文件時(shí),只能指定一個(gè)數(shù)據(jù)源, 并且我們?cè)陂_發(fā)到發(fā)布過程中最少有三套環(huán)境,dev, test, prod. 如果按照kettle只能指定一個(gè)數(shù)據(jù)源,對(duì)應(yīng)的kjb/ktr文件就得要三套,首先不說麻煩, 更容易導(dǎo)致出錯(cuò),測(cè)試發(fā)到生產(chǎn)的可能.所以我們要解決這個(gè)痛點(diǎn), 就是可以根據(jù)我們配置三套環(huán)境的數(shù)據(jù)源,然后根據(jù)環(huán)境變量自己去更換數(shù)據(jù)源.

1.踩坑

在查詢資料((#.#)其實(shí)就是百度)的過程中,找到如下方法,(以kjb文件為例):
注意該方法為錯(cuò)誤方法
踩坑點(diǎn):在將kjb的路徑傳到JobMeta里初始化時(shí)候就kettle就開始連接數(shù)據(jù)庫了,導(dǎo)致我在測(cè)試環(huán)境一直報(bào)數(shù)據(jù)庫連接錯(cuò)誤(作者公司test/dev環(huán)境是隔開的本地是沒辦法連接測(cè)試環(huán)境)

public static Job executeKjb(String kjbPath, Map<String, String> param) {
        log.info("execute krt, ktrPath={}, param={}", kjbPath, param);
        Job job;
        try {

            KettleEnvironment.init();
            //初始化job路徑
            JobMeta jobMeta = new JobMeta(kjbPath, null);
            //替換db鏈接信息(這里主要設(shè)置kjb文件里面的數(shù)據(jù)庫連接信息)
           setDBLinkInfo(KETTLE_DB_PARAM_LIST, jobMeta.getDatabases());
            job = new Job(null, jobMeta);
            //krt文件更換連接信息(修改kjb引用ktr文件里的數(shù)據(jù)庫連接信息)
           Map<JobEntryCopy, JobEntryTrans> activeJobEntryTransformations = job.getActiveJobEntryTransformations();
           activeJobEntryTransformations.entrySet().forEach(jobEntryCopyJobEntryTransEntry -> {
               DatabaseMeta[] connections = jobEntryCopyJobEntryTransEntry.getValue().getUsedDatabaseConnections();
               setDBLinkInfo(KETTLE_DB_PARAM_LIST, Arrays.asList(connections));
            });
            //初始化job參數(shù)歹垫,腳本中獲取參數(shù)值:${variableName}
            if (!CollectionUtils.isEmpty(param)) {
                job.injectVariables(param);
            }
            job.start();
            job.waitUntilFinished();
        } catch (Exception e) {
            log.error("kbj文件執(zhí)行失敗", e);
            throw new ServiceException(".kjb文件執(zhí)行失敗", e);
        }
        return job;
    }

//dbParams自己配置的各環(huán)境的數(shù)據(jù)庫連接信息
private static void setDBLinkInfo(List<KettleDBParams.KettleDBParam> dbParams, List<DatabaseMeta> databases) {
        //設(shè)置DB參數(shù)
        if (!CollectionUtils.isEmpty(dbParams)) {
            Map<String, KettleDBParams.KettleDBParam> connNameGroupMap = dbParams.stream()
                    .collect(Collectors.toMap(KettleDBParams.KettleDBParam::getConnName, e -> e, (o, o2) -> o));
            for (DatabaseMeta databaseMeta : databases) {
                KettleDBParams.KettleDBParam dbParam = connNameGroupMap.get(databaseMeta.getName());
                if (dbParam == null) {
                    throw new ServiceException(String.format("執(zhí)行.ktr失敗, 未發(fā)現(xiàn)數(shù)據(jù)庫連接【%s】", dbParam.getConnName()));
                }
                log.info("kettle數(shù)據(jù)庫使用的驅(qū)動(dòng)為:{}", databaseMeta.getDriverClass());
                //連接地址
                databaseMeta.setHostname(dbParam.getHost());
                //數(shù)據(jù)庫名稱
                databaseMeta.setDBName(dbParam.getDbName());
                //端口
                databaseMeta.setDBPort(dbParam.getPort());
                //用戶
                databaseMeta.setUsername(dbParam.getUsername());
                //密碼
                databaseMeta.setPassword(dbParam.getPassword());
            }
        } else {
            throw new ServiceException("需要替換的數(shù)據(jù)庫參數(shù)為空!!!");
        }
    }

2.正確修改方式(可能是之一)

作者用的是JNDI方式連接數(shù)據(jù)源

圖片.png

配置文件:
kettle解壓目錄/simple-jndi/jdbc.properties
以下是配置文件內(nèi)原本內(nèi)容
/前面就是JNDI名稱,上圖紅框內(nèi)容, 按照自己需要配置進(jìn)行配置即可

SampleData/type=javax.sql.DataSource
SampleData/driver=org.h2.Driver
SampleData/url=jdbc:h2:file:samples/db/sampledb;IFEXISTS=TRUE
SampleData/user=PENTAHO_USER
SampleData/password=PASSWORD
SampleDataAdmin/type=javax.sql.DataSource
SampleDataAdmin/driver=org.h2.Driver
SampleDataAdmin/url=jdbc:h2:file:samples/db/sampledb;IFEXISTS=TRUE
SampleDataAdmin/user=PENTAHO_ADMIN
SampleDataAdmin/password=PASSWORD
Quartz/type=javax.sql.DataSource
Quartz/driver=org.hsqldb.jdbcDriver
Quartz/url=jdbc:hsqldb:hsql://localhost/quartz
Quartz/user=pentaho_user
Quartz/password=password
Hibernate/type=javax.sql.DataSource
Hibernate/driver=org.hsqldb.jdbcDriver
Hibernate/url=jdbc:hsqldb:hsql://localhost/hibernate
Hibernate/user=hibuser
Hibernate/password=password
Shark/type=javax.sql.DataSource
Shark/driver=org.hsqldb.jdbcDriver
Shark/url=jdbc:hsqldb:hsql://localhost/shark
Shark/user=sa
Shark/password=
PDI_Operations_Mart/type=javax.sql.DataSource
PDI_Operations_Mart/driver=org.postgresql.Driver
PDI_Operations_Mart/url=jdbc:postgresql://localhost:5432/hibernate?searchpath=pentaho_operations_mart
PDI_Operations_Mart/user=hibuser
PDI_Operations_Mart/password=password
live_logging_info/type=javax.sql.DataSource
live_logging_info/driver=org.postgresql.Driver
live_logging_info/url=jdbc:postgresql://localhost:5432/hibernate?searchpath=pentaho_dilogs
live_logging_info/user=hibuser
live_logging_info/password=password

3.正確方式-java項(xiàng)目中運(yùn)用

我們只需要根據(jù)dev/test/prod各個(gè)環(huán)境去讀取相應(yīng)的jdbc.properties文件即可
然后我們?cè)谥纊ettle是如何讀取到j(luò)dbc.properties文件,
源碼如下:

//在init的時(shí)候就會(huì)設(shè)置讀取
KettleEnvironment.init();
↓ 一直點(diǎn)下去init方法
public static void init(List<PluginTypeInterface> pluginClasses, boolean simpleJndi) throws KettleException {
        SettableFuture ready;
        if (initialized.compareAndSet((Object)null, ready = SettableFuture.create())) {
            System.setProperties(ConcurrentMapProperties.convertProperties(System.getProperties()));

            try {
                if (!KettleClientEnvironment.isInitialized()) {
                    KettleClientEnvironment.init();
                }
                //這里是加載JNDI方式的方法
                if (simpleJndi) {
                    JndiUtil.initJNDI();
                }

                pluginClasses.forEach(PluginRegistry::addPluginType);
                PluginRegistry.init();
                KettleVariablesList.init();
                initLifecycleListeners();
                ready.set(true);
            } catch (Throwable var5) {
                ready.setException(var5);
                throw var5 instanceof KettleException ? (KettleException)var5 : new KettleException(var5);
            }
        } else {
            ready = (SettableFuture)initialized.get();

            try {
                ready.get();
            } catch (Throwable var4) {
                throw new KettleException(var4);
            }
        }

    }
↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓點(diǎn)進(jìn) JndiUtil.initJNDI();
 public static void initJNDI() throws KettleException {
//加載jdbc.properties的路徑是從Const.JNDI_DIRECTORY靜態(tài)變量里獲取, 如果為空就加載kettle文件內(nèi)數(shù)據(jù)源
        String path = Const.JNDI_DIRECTORY;
        if (path == null || path.equals("")) {
            try {
                File file = new File("simple-jndi");
                path = file.getCanonicalPath();
            } catch (Exception var2) {
                throw new KettleException("Error initializing JNDI", var2);
            }

            Const.JNDI_DIRECTORY = path;
        }

        System.setProperty("java.naming.factory.initial", "org.osjava.sj.SimpleContextFactory");
        System.setProperty("org.osjava.sj.root", path);
        System.setProperty("org.osjava.sj.delimiter", "/");
    }

那我們的思路就來了, 我們?cè)谡{(diào)用kettle文件之前賦值給Const.JNDI_DIRECTORY靜態(tài)變量我們自己的路徑就好了,那么下面就是我自己的加載方式
注意jdbc.properties文件名不要改, 應(yīng)該默認(rèn)讀取的,作者沒有往深里看

import com.xxxx.actuator.exception.ServiceException;
import lombok.extern.slf4j.Slf4j;
import org.pentaho.di.core.Const;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;
import java.io.File;

/**
 * 初始化kettle讀取數(shù)據(jù)源配置文件地址
 * 我是把jdbc.properties配置放到服務(wù)器上的,springboot打成jar讀取文件太麻煩了,
 *以下只是給一個(gè)啟發(fā), 具體如何讀取按照自己項(xiàng)目需求.
 */
@Component
@Slf4j
public class JndiInit {
    @Value("${spring.profiles}")
    private String evn;

    @PostConstruct
    public void init() {
        try {
            String fileSeparator = System.getProperty("file.separator");
            //開發(fā)環(huán)境獲取根目錄下文件路徑
            if ("dev".equals(evn)) {
                Const.JNDI_DIRECTORY = new File("").getCanonicalPath() + fileSeparator + "jndi" + fileSeparator + evn;

            } else if ("test".equals(evn) || "prod".equals(evn)) {
                //測(cè)試/生產(chǎn)環(huán)境
                Const.JNDI_DIRECTORY = "/home/service/app/executor/tmp" + fileSeparator + "jndi" + fileSeparator + evn;
            }
        } catch (Exception e) {
            throw new ServiceException("初始化jndi地址失敗", e);
        }
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子镐作,更是在濱河造成了極大的恐慌覆旱,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,042評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異抬纸,居然都是意外死亡帮掉,警方通過查閱死者的電腦和手機(jī)弦悉,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來旭寿,“玉大人警绩,你說我怎么就攤上這事≈殉疲” “怎么了肩祥?”我有些...
    開封第一講書人閱讀 156,674評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長缩膝。 經(jīng)常有香客問我混狠,道長,這世上最難降的妖魔是什么疾层? 我笑而不...
    開封第一講書人閱讀 56,340評(píng)論 1 283
  • 正文 為了忘掉前任将饺,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘予弧。我一直安慰自己刮吧,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,404評(píng)論 5 384
  • 文/花漫 我一把揭開白布掖蛤。 她就那樣靜靜地躺著杀捻,像睡著了一般。 火紅的嫁衣襯著肌膚如雪蚓庭。 梳的紋絲不亂的頭發(fā)上致讥,一...
    開封第一講書人閱讀 49,749評(píng)論 1 289
  • 那天,我揣著相機(jī)與錄音器赞,去河邊找鬼垢袱。 笑死港柜,一個(gè)胖子當(dāng)著我的面吹牛请契,可吹牛的內(nèi)容都是我干的潘懊。 我是一名探鬼主播,決...
    沈念sama閱讀 38,902評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼授舟,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼救恨!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起释树,我...
    開封第一講書人閱讀 37,662評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎奢啥,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體桩盲,經(jīng)...
    沈念sama閱讀 44,110評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡寂纪,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了赌结。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片捞蛋。...
    茶點(diǎn)故事閱讀 38,577評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡柬姚,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出量承,到底是詐尸還是另有隱情穴店,我是刑警寧澤拿穴,帶...
    沈念sama閱讀 34,258評(píng)論 4 328
  • 正文 年R本政府宣布,位于F島的核電站贞言,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏该窗。R本人自食惡果不足惜蚤霞,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,848評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望昧绣。 院中可真熱鬧,春花似錦拖刃、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,726評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽均函。三九已至菱涤,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間粘秆,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,952評(píng)論 1 264
  • 我被黑心中介騙來泰國打工殷勘, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留陋气,地道東北人劳吠。 一個(gè)月前我還...
    沈念sama閱讀 46,271評(píng)論 2 360
  • 正文 我出身青樓巩趁,卻偏偏與公主長得像淳附,于是被迫代替她去往敵國和親蠢古。 傳聞我的和親對(duì)象是個(gè)殘疾皇子奴曙,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,452評(píng)論 2 348