is there a way of calling external scripts with liquibase in a staged manner?
I am looking for something similar to flyway callbacks to call external scripts like sh: https://flywaydb.org/documentation/callbacks.html
Example:
1) pre-migration: run sh script 1
2) run migration
3) post-migration: run sh script 2
4) If migration fails: run sh script 3
Basically a staged mechanism of calling external scripts as part of the migration steps.
I would appreciate your feedback.
Thank you
Tobi
You can run external programs (including sh scripts) with http://www.liquibase.org/documentation/changes/execute_command.html
If you want to run something before and after each migration run, you need to structure your master changelog such way:
<changeSet id="preMigration" runAllways="true">
...pre migration
</changeSet>
...all your migrations are here
<changeSet id="postMigration" runAllways="true">
...post migration
</changeSet>
And I'm not sure that there is error handler.
Thank you dbf
I was able to run a bat file using liquibase and pass parameters into it:
<property name="my_param_name" value="myValue"/>
<changeSet author="tobi" id="preMigration" >
<executeCommand executable="C:\projects\lbdemo\trunk\mybatfile.bat">
<arg value="Constant: ${my_param_name}"/>
</executeCommand>
</changeSet>
C:\projects\lbdemo\trunk\mybatfile.bat :
#echo off
if not exist "C:\Test\" mkdir C:\Test
set arg1=%1
(echo %arg1%) > C:\Test\EmptyFile.txt
mybatfile.bat will create an EmptyFile.txt and write the my_param_name value into it.
Related
We have been using Liquibase successfully for about six months. I'm moving to a new CI/CD pipeline using CircleCI and running into an error when running liquibase update over SSH.
Here's the command (after many iterations and much exploration of Liquibase documentation):
ssh $SSH_USER#$TEST_JOB_SSH_HOST "cd /var/www/html/liquibase ; liquibase --url=jdbc:$TEST_DB_URL/$TEST_DB_SCHEMA?user=$TEST_DB_USERNAME --username=$TEST_DB_USERNAME --password="\""$TEST_DB_PASSWORD"\"" --changelog-file=cl-main.xml --search-path=.,./ update --log-level 1"
The result:
However, the file does exist and can be seen here:
It was successfully executed several months ago using our old approach. Now I think Liquibase is just parsing files and somehow failing, likely because it's running from a different directory.
Here's a snippet from the changeset file:
<sqlFile dbms="mysql, mariadb"
encoding="UTF-8"
endDelimiter=";"
path="/../data/regional_integration_details-ingest_day-01.sql"
relativeToChangelogFile="true"
splitStatements="true"
stripComments="true"/>
I think the issue is the leading slash.
The command I pasted above was based on reviewing this help document: https://docs.liquibase.com/concepts/changelogs/how-liquibase-finds-files.html
I'm struggling with the proper syntax to include in the --search-path parameter -- if that's even the correct parameter -- to make this work.
The nuclear option (yet to be tested) is to update all of our changesets, removing the leading slash. I'd prefer not to go that route.
Suggestions?
Edit 1
Updating to mention that the first four changesets are parsed successfully. They have path values like ../dirname/sqlscript_00.sql. Liquibase chokes on the first script with /../dirname/sqlscript_01.sql.
Also, we have no problems running Liquibase in local development, when we cd to /var/www/html/liquibase in our Docker containers and execute the liquibase update command.
Edit 2
Having CircleCI SSH directly into the server doesn't work, as it doesn't carry the variables over with it.
Passing the commands via SSH preserves those variables.
Liquibase removed support for absolute paths in v4.x.
I am using the following command to generate the change log
liquibase --schemas=test_schema --changelog-file=changelog.xml generateChangeLog
My liquibase.properties file is following:
url: jdbc:postgresql://xx.xxx.xx.xx:5432/test_db
driver: org.postgresql.Driver
username: testuser
password: test
logFile: liquibase.log
logLevel: debug
defaultSchemaName= test_schema
outputDefaultSchema= true
The output in the changelog.xml doesnt have a schemaName, I have multiple schemas in my database so I need schemaname in the changelog as well.
Output :
<changeSet author="user1 (generated)" id="xxxxxxxxxxxxx-1">
<createTable tableName="hello_world">
<column name="msg" type="CHAR(60)"/>
</createTable>
</changeSet>
Should be test_schema.hello_world or schemaname should be defined. Any suggestions what I am doing wrong?
Try to append currentSchema= to your db url
e.g.
command line:
--url="jdbc:postgresql://DB_HOST:DB_PORT/DB_NAME?currentSchema=SCHEMA_NAME"
liquibase.properties
url="jdbc:postgresql://DB_HOST:DB_PORT/DB_NAME?currentSchema=SCHEMA_NAME"
then call
liquibase <other command line params not part of liquibae.properties> generateChangeLog
According to this doc, the --schemas=<name1, name2> specifies database schemas you want to include. You might want to list out the schemas you want to include. Or you can simply use the liquibase --changeLogFile=dbchangelog.xml generateChangeLog command.
I am running liquibase update from command line.
When there is an error, I would like to log the error into a text file.
I tried the --logFile option with update command but it didn't work.
How do I write the liquibase errors into a file?
Thanks,
Radha
Normally you can just use the redirect > shell command. Like liquibase update > log.txt
The documentation for the <sqlFile> custom refactoring tag says that the classpath will be searched for the file.
However I cannot get it to find my file, despite it being in the classpath.
The changeset element is:
<changeSet author="rebecca" id="9.1 - LoanIQ GoLive">
<comment>No rollback script exists</comment>
<sqlFile path="v9.1-loaniqgolive.sql"/>
</changeSet>
The ant task is:
<updateDatabase
changeLogFile="#{changelog}"
driver="com.microsoft.sqlserver.jdbc.SQLServerDriver"
url="jdbc:sqlserver://${database.host}:${database.port};databaseName=${database.name}"
username="${database.user}"
password="${database.password}"
promptOnNonLocalDatabase="false"
dropFirst="false"
classpathref="liquibase.path"/>
The build reports that the liquibase path is:
[echo] The liquibase path = D:\Program Files\Jenkins\jobs\Deploy GMM\workspace\app\build\database;D:\Program Files\Jenkins\jobs\Deploy GMM\workspace\app\build\tools\liquibase\liquibase.jar;D:\Program Files\Jenkins\jobs\Deploy GMM\workspace\app\build\lib\sqljdbc.jar
I have confirmed that v9.1-loaniqgolive.sql is definitely in the path (1st element).
Yet I get this error:
D:\Program Files\Jenkins\jobs\Deploy GMM\workspace\app\build\ant\functions\db.xml:56: liquibase.exception.ChangeLogParseException: Invalid Migration File: <sqlfile path=v9.1-loaniqgolive.sql> -Unable to read file
at liquibase.parser.core.xml.XMLChangeLogSAXParser.parse(XMLChangeLogSAXParser.java:132)
My config looks OK, but it doesn't work. How can I solve this or debug further?
One quick suggestion I have is to run Ant with -verbose and -debug command line parameters. This puts out a lot of logging but generally helps get you closer to the source of the problem.
I will try to create an equivalent Ant task against one of my dbs and respond with any updates.
I have a .sql file which contain create script for 20 tables.I am running this through ant.
the Ant Script is
<target name="CompleteDBRollCreate" >
<sql classpath="D:\lib\ojdbc14.jar" driver="oracle.jdbc.driver.OracleDriver"
url="jdbc:oracle:thin:#12326:1521:orcl"
userid="scott" password="tiger"
src="D:\dropalltables.sql" />
</target>
but there is no logging in ant so I can't know if the SQL file was executed correctly. Can any one please suggest how to do logging into a separate localfile/on the ant console. I just want to know where I need to change the .sql file or the ant script.
There is logging in ant. Check out Running Apache Ant - Command Line:
-logfile use given file for log
-logfile will give you jdbc jdbc exceptions and all the noise around that, which can be useful, but I would also recommend spooling in the sql file as well as it looks like you are going to have multiple statements in the script. So I'd say you should use the -logfile option and also alter your script to spool any ORA errors you might encounter if you have multiple statements.
spool test.out
create table foo (testCol varchar2(2));
drop table foo;
spool off