About execute variable
execute
is a Jinja variable that returns True when dbt is in "execute" mode.
When you execute a dbt compile
or dbt run
command, dbt:
- Reads all of the files in your project and generates a manifest comprised of models, tests, and other graph nodes present in your project. During this phase, dbt uses the
ref
andsource
statements it finds to generate the DAG for your project. No SQL is run during this phase, andexecute == False
. - Compiles (and runs) each node (eg. building models, or running tests). SQL is run during this phase, and
execute == True
.
Any Jinja that relies on a result being returned from the database will error during the parse phase. For example, this SQL will return an error:
models/order_payment_methods.sql
1 {% set payment_method_query %}
2 select distinct
3 payment_method
4 from {{ ref('raw_payments') }}
5 order by 1
6 {% endset %}
7
8 {% set results = run_query(payment_method_query) %}
9
10 {# Return the first column #}
11 {% set payment_methods = results.columns[0].values() %}
The error returned by dbt will look as follows:
Encountered an error:
Compilation Error in model order_payment_methods (models/order_payment_methods.sql)
'None' has no attribute 'table'
This is because line #11 in the earlier code example ({% set payment_methods = results.columns[0].values() %}
) assumes that a tableIn simplest terms, a table is the direct storage of data in rows and columns. Think excel sheet with raw values in each of the cells. has been returned, when, during the parse phase, this query hasn't been run.
To work around this, wrap any problematic Jinja in an {% if execute %}
statement:
models/order_payment_methods.sql
{% set payment_method_query %}
select distinct
payment_method
from {{ ref('raw_payments') }}
order by 1
{% endset %}
{% set results = run_query(payment_method_query) %}
{% if execute %}
{# Return the first column #}
{% set payment_methods = results.columns[0].values() %}
{% else %}
{% set payment_methods = [] %}
{% endif %}
0