jsonapi-rb / jsonapi-serializable

Conveniently build and efficiently render JSON API resources.
http://jsonapi-rb.org
MIT License
45 stars 35 forks source link

Block for conditional field is always called #49

Open ezekg opened 7 years ago

ezekg commented 7 years ago

Example

relationship :user, unless: -> { @object.user.nil? } do
  link :related do
    @url_helpers.v1_user_path @object.user
  end
end

Expected outcome

The relationship block is not called when the conditional fails.

Actual outcome

The relationship block is called regardless of the conditional failing; only the display of the serialized relationship is changed. This requires additional checks within the block, which makes the code less readable (especially for nested resource routes) and duplicates the condition:

relationship :user, unless: -> { @object.user.nil? } do
  link :related do
    @url_helpers.v1_user_path @object.user if @object.user.present?
  end
end

Solution

Do not call the block if the condition fails.


Would you be open to a pull request?

beauby commented 7 years ago

@ezekg You are right – at the moment links are built eagerly (only the related objects are constructed lazily). However, a simple fix for your use-case (that would be in-line with the spirit of the spec) would be to expose the related user via some nested route (like /posts/3/user), rather than the canonical resource url (/users/:id).

Regarding the former, I am currently refactoring that part to avoid eagerly building stuff, so it should be fixed for the next release.

beauby commented 7 years ago

@ezekg See https://github.com/json-api/json-api/issues/1134 and related issues for an overview of why exposing the related link as /posts/3/user is the most flexible solution.

ezekg commented 7 years ago

As much as I'd like to implement that, at the moment it brings in too much overhead into developing the API. I'd need to create new routes/controllers for each relationship, and write tests for them.

Appreciate the guidance though. I'm going to create a ticket as a reminder to come back to this.

beauby commented 7 years ago

@ezekg Would you mind writing a regression test for this?

ezekg commented 7 years ago

Definitely. I'll try and do that this weekend.

dylanlewis89 commented 7 years ago

Out of curiosity, does the library currently support specifying any conditional rendering of attributes in the serializer via the exposed instance variables?

For example:

class SerializableBar < JSONAPI::Serializable::Resource
  type 'bars'

  attribute :always_show_me
  attribute :conditionally_show_me, unless: -> { @authenticated }
end

This requested feature was the closest I could find to this and just wanted to see if I was missing anything obvious that already existed in the library.

beauby commented 7 years ago

@dylanlewis89 Yes it does, you just have to call extend JSONAPI::Serializable::Resource::KeyFormat in your serializer in order to enable it.

beauby commented 7 years ago

@dylanlewis89 Sorry, I obviously meant JSONAPI::Serializable::Resource::ConditionalFields rather than JSONAPI::Serializable::Resource::KeyFormat.

dylanlewis89 commented 7 years ago

Thanks @beauby for the quick and helpful response!

For anyone tackling this issue in the future, the 0.2.1 release introduces the extend JSONAPI::Serializable::Resource::ConditionalFields syntax. Earlier versions use prepend instead of extend

beauby commented 7 years ago

@ezekg So if you're still interested in fixing this, it's just a matter of delaying the instance_evaling here until either as_jsonapi or related_resources are called.

ezekg commented 7 years ago

Hey, I just updated my API to use the latest version of jsonapi-rb/rails so I can dig into fixing this now! I was trying to hold off until I could get around to updating, as I was still on v0.1.1. 🙂