How to get foreign key values with getattr from models

Question:

I have a model Project and i am getting the attributes of that with the following instr

attr = getattr(project, 'id', None)

project is the instance, id is the field and None is the default return type.

My question is: what if I want to get the Foreign Key keys with this?

Get customer name

project.customer.name
How to get customer name with the above condition?

Already Tried

if callable(attr):
     context[node][field] = '%s' % attr()

Current Code

context = {'project': {}}
fields = ('id', 'name', 'category', 'created_by', customer)

for field in fields:
    attr = getattr(project, field, None)
        if callable(attr):
            context['project'][field] = '%s' % attr()
        else:
            context['project'][field] = attr

i need to adjust customer object here. that i can give something like customer__name or customer.name in my fields and it get populated with the name of the customer, but i am not sure.

Asked By: A.J.

||

Answers:

You can do something like follows:

def get_repr(value): 
    if callable(value):
        return '%s' % value()
    return value

def get_field(instance, field):
    field_path = field.split('.')
    attr = instance
    for elem in field_path:
        try:
            attr = getattr(attr, elem)
        except AttributeError:
            return None
    return attr

for field in fields:
    context['project'][field] = get_repr(get_field(project, field))
Answered By: alko

Here’s an equivalent recursive getattr

def get_foreign_key_attr(obj, field: str):
    """Get attr recursively by following foreign key relations

    For example ...
    get_foreign_key_attr(
        <Station: Spire operational 1m>, "idproject__idcountry__name"
    )
    ... splits "idproject__idcountry__name" into ["idproject", "idcountry", "name"]
    & follows finds the value of each foreign key relation
    """
    fields = field.split("__")
    if len(fields) == 1:
        return getattr(obj, fields[0], "")
    else:
        first_field = fields[0]
        remaining_fields = "__".join(fields[1:])
        return get_foreign_key_attr(getattr(obj, first_field), remaining_fields)
Answered By: rdmolony