Literal scalar expands to a dictionary

For example, given "val1: foo"

    - set_fact:
        _var1: |
          {"key1": "{{ val1 }}"}
        _var2: |
          {"key1": "foo"}
    - debug:
        var: _var1|type_debug
    - debug:
        var: _var2|type_debug

gives

  _var1|type_debug: dict
  _var2|type_debug: AnsibleUnicode

*_var1* expands to a dictionary but *_var2* expands to a string. Is it
a feature or a bug?

Thank you,

#1 from this response explains why this happens:

https://github.com/ansible/ansible/issues/34595#issuecomment-356091161

If the value contains a variable, it is processed through Templar.template
a. Templar.template is powered by jinja2, and historically only has the ability to return strings
b. Because jinja2 has historically only had the ability to template strings, we do some magic to try and convert strings that look like python data structures to python data structures
c. The string now becomes a python dictionary

You can work around this, although it’s a bit more complex:

  • set_fact:
    _var1: |
    {{ (‘{“key1”: "’ ~ val1 ~ ‘"}’) | string }}

See https://docs.ansible.com/ansible-core/2.12/reference_appendices/config.html#string-type-filters

There are likely some other variants on my example.

Why don't you try this magic consistently both on jinja2 and
non-jinja2 expressions?

Thank you,

I believe that was explained. The “magic” as you call it, only affects jinja2 templating, because historically jinja2 could only ever generate strings, but the intention was often to generate python data types.

We don’t do that for things that don’t go through templating, because the author can define it explicitly as the type they want.

Thank you. I try to understand the logic. In summary (up till now),

  "The magic only affects jinja2 templating; to try and convert
  strings that look like python data structures to python data
  structures."

How do you decide what looks like python data structure? For example,
only the second string below is converted to dictionary, despite the
fact that both are valid YAML dictionaries

    - set_fact:
        _var3: '{"key1": {{ val1 }}}'
        _var4: '{"key1": "{{ val1 }}"}'
    - debug:
        var: _var3|type_debug
    - debug:
        var: _var3|from_yaml|type_debug
    - debug:
        var: _var4|type_debug

gives

  _var3|type_debug: str
  _var3|from_yaml|type_debug: dict
  _var4|type_debug: dict