A Db2 dialect plugin for SQLGlot - a powerful SQL parser, transpiler, and optimizer.
- Python: 3.9 or higher
- SQLGlot: 30.0.0 or higher
- Full Db2 SQL syntax support
- Cross-dialect transpilation (Db2 ↔ PostgreSQL, MySQL, Snowflake, etc.)
- Type mapping (BOOLEAN → SMALLINT, NCHAR/NVARCHAR support, etc.)
- Db2-specific functions (POSSTR, VARCHAR_FORMAT, DAYOFWEEK, DAYOFYEAR)
- FETCH FIRST syntax support
- NULL ordering support
All tests passing: 6 tests, 28 subtests
$ python3 -m pytest tests/test_db2_dialect.py -v
============================= test session starts ==============================
tests/test_db2_dialect.py::TestDB2::test_db2 PASSED [ 16%]
tests/test_db2_dialect.py::TestDB2::test_nchar_nvarchar_transpilation PASSED [ 33%]
tests/test_db2_dialect.py::TestDB2::test_null_ordering PASSED [ 50%]
tests/test_db2_dialect.py::TestDB2::test_strip_modifiers PASSED [ 66%]
tests/test_db2_dialect.py::TestDB2::test_typed_division PASSED [ 83%]
tests/test_db2_dialect.py::TestDB2::test_variable_tokens PASSED [100%]
==================== 6 passed, 28 subtests passed in 0.28s =====================The test suite validates:
- ✅ Basic SQL: SELECT, INSERT, UPDATE, DELETE, CREATE/DROP TABLE
- ✅ Type conversions: INTEGER→INT, NCHAR→GRAPHIC, NVARCHAR→VARGRAPHIC, DBCLOB→CLOB
- ✅ Functions: POSSTR, VARCHAR_FORMAT, DAYOFWEEK, DAYOFYEAR, GREATEST, LEAST
- ✅ Boolean handling: TRUE/FALSE → 1/0
- ✅ Date/Time: CURRENT DATE, CURRENT TIMESTAMP, DATEDIFF→DAYS
- ✅ FETCH FIRST: Pagination with FETCH FIRST n ROWS ONLY
- ✅ OFFSET: OFFSET n ROWS syntax
- ✅ NULL ordering: NULLS FIRST, NULLS LAST
- ✅ Joins: INNER, LEFT, RIGHT joins
- ✅ Aggregations: COUNT, SUM, AVG, MIN, MAX
- ✅ Subqueries & CTEs: WITH clause support
- ✅ CASE expressions: Simple and searched CASE
- ✅ Operators: IN, BETWEEN, LIKE, IS NULL
- ✅ Set operations: UNION, UNION ALL
- ✅ Variable tokens: @var syntax
- ✅ Typed division: Proper numeric division handling
cd db2-sqlglot-dialect
pip install -e .pip install db2-sqlglot-dialectOnce installed, the Db2 dialect is automatically discovered by SQLGlot:
from sqlglot import transpile
# Transpile from PostgreSQL to Db2 (LIMIT → FETCH FIRST)
result = transpile(
"SELECT * FROM table1 LIMIT 10",
read="postgres",
write="db2"
)
print(result[0])
# Output: SELECT * FROM table1 FETCH FIRST 10 ROWS ONLY
# Transpile from Db2 to PostgreSQL
# Note: FETCH FIRST is preserved (PostgreSQL supports it natively - SQL standard)
result = transpile(
"SELECT * FROM table1 FETCH FIRST 10 ROWS ONLY",
read="db2",
write="postgres"
)
print(result[0])
# Output: SELECT * FROM table1 FETCH FIRST 10 ROWS ONLY
# Both LIMIT and FETCH FIRST work in PostgreSQL
# If you need LIMIT specifically, use it in the source query
result = transpile(
"SELECT * FROM table1 LIMIT 10",
read="db2",
write="postgres"
)
print(result[0])
# Output: SELECT * FROM table1 LIMIT 10- Standard types: INTEGER, BIGINT, SMALLINT, DECIMAL, VARCHAR, CHAR, DATE, TIMESTAMP, CLOB, BLOB, BOOLEAN
- Db2-specific types: GRAPHIC, VARGRAPHIC, DBCLOB, DATALINK, ROWID, DECFLOAT, XML
When transpiling from other databases to Db2:
- ✅ TEXT → CLOB
- ✅ BYTEA/BINARY → BLOB
- ✅ TINYINT → SMALLINT
- ✅ TIMESTAMPTZ → TIMESTAMP
⚠️ SERIAL/BIGSERIAL → Preserved as-is (may need manual conversion)⚠️ LONGTEXT/MEDIUMTEXT → Preserved as-is (may need manual conversion)
Note: Unsupported or dialect-specific constructs may be preserved as-is during transpilation. This behavior prioritizes syntax preservation over potentially lossy or incorrect transformations. Users should review and adjust preserved types as needed for their specific Db2 version and requirements.
POSSTR(haystack, needle)- String positionVARCHAR_FORMAT(timestamp, format)- Time to string conversionDAYOFWEEK(date)- Extract day of weekDAYOFYEAR(date)- Extract day of yearCURRENT DATE,CURRENT TIMESTAMP- Date arithmetic with
+and-
FETCH FIRST n ROWS ONLYsyntaxOFFSET n ROWSsyntax- NULL ordering (
NULLS FIRST,NULLS LAST) - Typed division
- Variable tokens (
@var)
This project includes two automated CI/CD workflows:
-
Unit Tests (
.github/workflows/unit-tests.yml)- Triggers: Push to main, Pull Requests, Manual dispatch
- Platforms: Ubuntu, macOS, Windows
- Python versions: 3.9, 3.10, 3.11, 3.12
- Features:
- Concurrency control (cancels outdated runs)
- Unit tests with coverage reporting
- Code quality checks (black, isort, ruff)
- Codecov integration
-
Build and Publish Release (
.github/workflows/build_release.yaml)- Triggers:
- Version tag push (for example,
v1.0.1) - Manual workflow dispatch (with choice: Test PyPI or PyPI, plus optional ref)
- Version tag push (for example,
- Features:
- Authorized release gating
- Builds distribution packages (wheel + sdist)
- Tests installation on multiple platforms before publish
- Publishes through GitHub environments (
test-pypi/pypi) - Supports approval-gated publishing through environment protection rules
- Triggers:
# Edit pyproject.toml
version = "1.0.1"
# Commit and push
git commit -am "Bump version to 1.0.1"
git push origin mainYou can trigger the release workflow in either of these ways:
git tag v1.0.1
git push origin v1.0.1This triggers .github/workflows/build_release.yaml.
- Go to Actions tab → Build and Publish Release
- Click "Run workflow"
- Select:
- Where to publish? →
test-pypiorpypi - Git ref → tag or branch (for example,
v1.0.1ormain)
- Where to publish? →
- Click "Run workflow"
The workflow:
- checks that the triggering GitHub user is authorized
- builds the package
- validates it with
twine check - tests wheel installation on supported platforms
- waits for environment approval if configured for
test-pypiorpypi - publishes to the selected package index
For Test PyPI:
pip install --index-url https://test.pypi.org/simple/ --extra-index-url https://pypi.org/simple/ db2-sqlglot-dialectFor PyPI:
pip install db2-sqlglot-dialect✅ Authorized releases only: Release workflow checks allowed GitHub users ✅ Build verification first: Package is built and validated before publish ✅ Install verification: Wheel is tested before publish ✅ Approval support: GitHub environments can require approval before publish ✅ Flexible triggering: Supports both version tags and manual dispatch
pytest tests/pytest tests/ -v --cov=db2_sqlglot --cov-report=term --cov-report=htmldb2-sqlglot-dialect/
├── db2_sqlglot/
│ ├── __init__.py
│ ├── dialect.py # Main dialect class
│ ├── generator.py # SQL generation logic
│ └── parser.py # SQL parsing logic
├── tests/
│ └── test_db2_dialect.py
├── pyproject.toml # Modern Python packaging (PEP 517/518/621)
└── README.md
Note: This project uses modern Python packaging with pyproject.toml only (no setup.py needed). Entry points are defined in pyproject.toml and work seamlessly with SQLGlot's plugin discovery system.
Contributions are welcome! Please feel free to submit a Pull Request.
Built on top of SQLGlot by Toby Mao.