Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support for expandable SerializerMethodFields #54

Open
J-Priebe opened this issue May 20, 2020 · 3 comments
Open

Support for expandable SerializerMethodFields #54

J-Priebe opened this issue May 20, 2020 · 3 comments

Comments

@J-Priebe
Copy link

Hello, I have a few large SerializerMethodFields on my serializer, which return iterables that are not a FK to my model. Something like this:

Class FooSerializer(FlexFieldsModelSerializer):
    things = SerializerMethodField()

    def get_things(self, instance):
        return ThingSerializer(get_things_related_to_this_instance(instance), many=True).data

As far as I know I cannot directly leverage drf-flex-fields to make this field expandable, because expandable_fields are a statically defined dict that takes a serializer class or tuple.

I have hacked around this by making these fields omitted by default, unless declared in the expand argument:

def __init__(self, *args, **kwargs):
    super().__init__(*args, **kwargs)

    expandable_method_fields = ('things', 'more_things')

    if '~all' in self._flex_options['expand'] or '*' in self._flex_options['expand']:
        return

    for field in expandable_method_fields:
        if (
            field not in self._flex_options['expand']
            and field not in self._flex_options['omit']
        ):
            self._flex_options['omit'].append(field)

It works well enough for my purposes, but I thought I'd pitch this as a feature request, in case it is helpful or others have solved the problem differently. Cheers, and thanks for the library!

@rsinger86
Copy link
Owner

I like the idea of supporting DRF's SerializerMethodField as an expandable field. One thing, though, is that it doesn't take keyword arguments for expand, omit, fields. This would only matter if you wanted dynamic behavior within that method.

class UserSerializer(serializers.ModelSerializer):
    friends = HyperlinkFIeld()

    class Meta:
        model = User
        expandable_fields = {'friends': serializer.SerializerMethodField}

    def get_friends(self, obj, expand, fields, omit):
        # serialize friends using custom logic

We could add a custom FlexSerializerMethodField - a subclass of SerializerMethodField that accepts those keyword arguments. Another possibility is making it a little smarter so it doesn't try to pass the keyword arguments if the expanded field is an instance of SerializerMethodField.

@vikrambombhi
Copy link

Is it not possible to do something like: FooSerializer.Meta.exclude = ("friends", ) and ask for it in query params if we want it /route?fields=friends ?

I think this breaks the model of expanding a field. Getting back the non expanded version of friends is different from not getting anything at all. I really like the idea of having something like FlexSerializerMethodField as @rsinger86 mentioned. Are there any plans on getting something like this implemented?

@michaelschem
Copy link
Contributor

One workaround that I've used is to convert your method field to an annotation on the queryset and expand on that. It also has the benefit of being much faster in most cases.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

4 participants