Advanced Jinja Techniques

Free Lesson

Advertisement

Advanced Jinja Techniques

Dispatch Architecture

Architecture Diagram
ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”
│                     JINJA DISPATCH ARCHITECTURE                             │
ā”œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”¤
│                                                                             │
│  ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”   │
│  │                    DISPATCH FLOW                                     │   │
│  │                                                                     │   │
│  │  Macro Call: {{ adapter.dispatch('macro_name', 'package') }}       │   │
│  │                                                                     │   │
│  │  1. Check project-level override                                    │   │
│  │  2. Check package-level implementation                              │   │
│  │  3. Check adapter-specific implementation                           │   │
│  │  4. Use default implementation                                      │   │
│  ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜   │
│                              │                                              │
│                              ā–¼                                              │
│  ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”   │
│  │                    DISPATCH EXAMPLE                                 │   │
│  │                                                                     │   │
│  │  {% macro safe_cast(column, type) %}                                │   │
│  │      {% set macro = adapter.dispatch('safe_cast', 'my_package')     │   │
│  │                           (column, type) %}                        │   │
│  │      {{ return(macro) }}                                            │   │
│  │  {% endmacro %}                                                     │   │
│  │                                                                     │   │
│  │  Implementations:                                                   │   │
│  │  ā”œā”€ā”€ default__safe_cast      (fallback)                            │   │
│  │  ā”œā”€ā”€ snowflake__safe_cast    (Snowflake-specific)                  │   │
│  │  ā”œā”€ā”€ bigquery__safe_cast     (BigQuery-specific)                   │   │
│  │  └── redshift__safe_cast     (Redshift-specific)                   │   │
│  ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜   │
│                                                                             │
ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜

Adapter Pattern

Architecture Diagram
ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”
│                     ADAPTER PATTERN IMPLEMENTATION                          │
ā”œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”¤
│                                                                             │
│  ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”   │
│  │                    ADAPTER HIERARCHY                                 │   │
│  │                                                                     │   │
│  │                    ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”                                 │   │
│  │                    │ BaseAdapter  │                                 │   │
│  │                    │   (dbt-core) │                                 │   │
│  │                    ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”¬ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜                                 │   │
│  │                           │                                         │   │
│  │           ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”¼ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”                        │   │
│  │           │               │               │                        │   │
│  │           ā–¼               ā–¼               ā–¼                        │   │
│  │    ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā” ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”            │   │
│  │    │  Snowflake   │ │  BigQuery    │ │  Redshift    │            │   │
│  │    │  Adapter     │ │  Adapter     │ │  Adapter     │            │   │
│  │    ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜ ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜            │   │
│  ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜   │
│                              │                                              │
│                              ā–¼                                              │
│  ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”   │
│  │                    MACRO DISPATCH                                   │   │
│  │                                                                     │   │
│  │  ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”   │   │
│  │  │  safe_cast(column, type)                                    │   │   │
│  │  │                                                             │   │   │
│  │  │  Snowflake: safe_cast({{ column }} as {{ type }})          │   │   │
│  │  │  BigQuery:  safe_cast({{ column }} as {{ type }})          │   │   │
│  │  │  Redshift:  cast({{ column }} as {{ type }})               │   │   │
│  │  │  Default:   cast({{ column }} as {{ type }})               │   │   │
│  │  ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜   │   │
│  ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜   │
│                                                                             │
ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜

Custom Macro Architecture

Architecture Diagram
ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”
│                     CUSTOM MACRO ARCHITECTURE                               │
ā”œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”¤
│                                                                             │
│  ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”   │
│  │                    MACRO LIBRARY STRUCTURE                           │   │
│  │                                                                     │   │
│  │  macros/                                                            │   │
│  │  ā”œā”€ā”€ general/                                                       │   │
│  │  │   ā”œā”€ā”€ string_utils.sql                                          │   │
│  │  │   ā”œā”€ā”€ date_utils.sql                                            │   │
│  │  │   └── math_utils.sql                                            │   │
│  │  ā”œā”€ā”€ cross_db/                                                      │   │
│  │  │   ā”œā”€ā”€ safe_cast.sql                                             │   │
│  │  │   ā”œā”€ā”€ date_trunc.sql                                            │   │
│  │  │   └── concat.sql                                                │   │
│  │  ā”œā”€ā”€ schema_tests/                                                  │   │
│  │  │   ā”œā”€ā”€ test_unique.sql                                           │   │
│  │  │   └── test_not_null.sql                                         │   │
│  │  └── generate/                                                      │   │
│  │      ā”œā”€ā”€ create_table.sql                                          │   │
│  │      └── merge.sql                                                 │   │
│  ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜   │
│                              │                                              │
│                              ā–¼                                              │
│  ā”Œā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”   │
│  │                    MACRO USAGE PATTERNS                              │   │
│  │                                                                     │   │
│  │  1. Direct call: {{ macro_name(args) }}                            │   │
│  │  2. Return value: {{ return(value) }}                              │   │
│  │  3. Call macro: {% call macro() %}...{% endcall %}                 │   │
│  │  4. Import: {% import 'macros/utils.sql' as utils %}              │   │
│  │  5. From import: {% from 'macros/utils.sql' import safe_cast %}   │   │
│  ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜   │
│                                                                             │
ā””ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”€ā”˜

Detailed Explanation

Advanced Jinja techniques in dbt enable sophisticated data transformation patterns, cross-database compatibility, and reusable code libraries.

Dispatch Pattern

The dispatch pattern allows macros to have database-specific implementations:

  1. Base implementation: Default fallback
  2. Adapter-specific: Database-optimized versions
  3. Package-level: Shared across projects
  4. Project-level: Custom overrides

Adapter Functions

dbt provides adapter functions for database-specific operations:

  • adapter.dispatch(): Route to correct implementation
  • adapter.resolve(): Resolve model references
  • adapter.get_relation(): Get database objects
  • adapter.create_schema(): Create database schemas
  • adapter.drop_relation(): Drop database objects

Macro Libraries

Organize macros into reusable libraries:

  1. General utilities: String, date, math functions
  2. Cross-database: Database-agnostic functions
  3. Schema tests: Reusable test definitions
  4. Generation macros: Code generation utilities

Advanced Patterns

  1. Recursive macros: Process hierarchical data
  2. Conditional logic: Dynamic SQL based on conditions
  3. Loop constructs: Process collections efficiently
  4. Context manipulation: Modify compilation context

Code Examples

Cross-Database Dispatch

-- macros/cross_db/safe_cast.sql
{% macro safe_cast(column, type) %}
    {% set macro = adapter.dispatch('safe_cast', 'dbt_utils')(column, type) %}
    {{ return(macro) }}
{% endmacro %}

{% macro default__safe_cast(column, type) %}
    cast({{ column }} as {{ type }})
{% endmacro %}

{% macro snowflake__safe_cast(column, type) %}
    safe_cast({{ column }} as {{ type }})
{% endmacro %}

{% macro bigquery__safe_cast(column, type) %}
    safe_cast({{ column }} as {{ type }})
{% endmacro %}

{% macro redshift__safe_cast(column, type) %}
    case 
        when {{ column }} ~ '^[0-9]+\.?[0-9]*$' then cast({{ column }} as {{ type }})
        else null
    end
{% endmacro %}

Advanced Macro with Call Block

-- macros/generate_merge.sql
{% macro generate_merge(target, source, unique_key, update_columns, insert_columns) %}
    {% call statement('merge') %}
        merge into {{ target }} as target
        using {{ source }} as source
        on {{ unique_key }}
        
        when matched then update set
            {% for col in update_columns %}
                {{ col }} = source.{{ col }}
                {% if not loop.last %},{% endif %}
            {% endfor %}
        
        when not matched then insert (
            {{ insert_columns | join(', ') }}
        )
        values (
            {% for col in insert_columns %}
                source.{{ col }}
                {% if not loop.last %},{% endif %}
            {% endfor %}
        )
    {% endcall %}
{% endmacro %}

Recursive Macro for Hierarchical Data

-- macros/generate_recursive_cte.sql
{% macro recursive_cte(cte_name, base_query, recursive_query, max_depth=10) %}
    with recursive {{ cte_name }} as (
        {{ base_query }}
        
        union all
        
        select
            {% for col in base_columns %}
                {{ cte_name }}_next.{{ col }}
                {% if not loop.last %},{% endif %}
            {% endfor %}
        from {{ cte_name }}
        inner join (
            {{ recursive_query }}
        ) {{ cte_name }}_next
        on {{ cte_name }}.id = {{ cte_name }}_next.parent_id
        where {{ cte_name }}.depth < {{ max_depth }}
    )
    
    select * from {{ cte_name }}
{% endmacro %}

Dynamic Column Generation

-- macros/generate_pivot.sql
{% macro pivot(source, group_by_columns, pivot_column, value_column, agg='sum') %}
    {% set pivot_values = run_query(
        "select distinct " ~ pivot_column ~ " from " ~ source ~ " order by 1"
    ).columns[0].values() %}
    
    select
        {{ group_by_columns | join(', ') }},
        {% for value in pivot_values %}
            {{ agg }}(case when {{ pivot_column }} = '{{ value }}' then {{ value_column }} end) as {{ value_column }}_{{ value | replace(' ', '_') | lower }}
            {% if not loop.last %},{% endif %}
        {% endfor %}
    from {{ source }}
    group by {{ group_by_columns | join(', ') }}
{% endmacro %}

Custom Test Macro

-- macros/schema_tests/test_freshness.sql
{% test freshness(model, column_name, interval, datepart) %}
    
    with source_data as (
        select
            max({{ column_name }}) as last_record,
            {{ dbt_utils.current_timestamp() }} as current_time
        from {{ model }}
    ),
    
    validation as (
        select
            last_record,
            current_time,
            {{ dbt_utils.datediff(
                "last_record",
                "current_time",
                datepart
            )}} as time_diff
        from source_data
    )
    
    select *
    from validation
    where time_diff > {{ interval }}

{% endtest %}

Advanced Dispatch with Package Override

-- macros/cross_db/date_trunc.sql
{% macro date_trunc(datepart, date) %}
    {% set macro = adapter.dispatch('date_trunc', 'dbt_utils')(datepart, date) %}
    {{ return(macro) }}
{% endmacro %}

{% macro default__date_trunc(datepart, date) %}
    date_trunc({{ datepart }}, {{ date }})
{% endmacro %}

{% macro snowflake__date_trunc(datepart, date) %}
    date_trunc({{ datepart }}, {{ date }})
{% endmacro %}

{% macro bigquery__date_trunc(datepart, date) %}
    date_trunc({{ date }}, {{ datepart }})
{% endmacro %}

{% macro redshift__date_trunc(datepart, date) %}
    date_trunc({{ datepart }}, {{ date }})
{% endmacro %}

{% macro postgres__date_trunc(datepart, date) %}
    date_trunc({{ datepart }}, {{ date }})
{% endmacro %}

Performance Metrics

PatternCompilation TimeUse Case
Direct callO(1)Simple macros
DispatchO(n)Cross-database
RecursiveO(depth)Hierarchical data
Dynamic columnsO(rows)Pivot operations
ImportO(1)Code reuse

Best Practices

  1. Use dispatch for cross-database compatibility
  2. Organize macros into logical directories
  3. Document macros with clear descriptions
  4. Test macros with different inputs
  5. Use return() to return values from macros
  6. Leverage call blocks for complex operations
  7. Import strategically to avoid circular dependencies
  8. Use adapter functions for database-specific operations

Advertisement

Need Expert dbt Help?

Get personalized tutoring, project support, or professional consulting.

Advertisement