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

Integrate with Active Model Attributes #410

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

seanpdoyle
Copy link

The schema { ... } interface pre-dates the Active Model Attributes API (defined as early as v5.2.0), but clearly draws inspiration from Active Record's Database Schema and Attribute casting (which was extracted into ActiveModel::Attributes).

However, the type information captured in schema { ... } blocks or assigned as Hash arguments to schema= is purely inert metadata.

Proposal

This commit aims to integrate with ActiveModel::Model and ActiveModel::Attributes. Through the introduction of both modules, subclasses of ActiveResource::Base can benefit from type casting attributes and constructing instances with default values.

This commit makes minimally incremental changes, prioritizing backwards compatibility. The reliance on #respond_to_missing? and #method_missing is left largely unchanged. Similarly, the Schema interface continues to provide metadata about its attributes through the Schema#attr method (instead of reading from
ActiveModel::Attributes#attribute_names or
ActiveModel::Attributes.attribute_types).

API Changes

To cast values to their specified types, declare the Schema with the :cast_values set to true.

class Person < ActiveResource::Base
  schema cast_values: true do
    integer 'age'
  end
end

p = Person.new
p.age = "18"
p.age # => 18

To configure inheriting resources to cast values, set the cast_values class attribute:

class ApplicationResource < ActiveResource::Base
  self.cast_values = true
end

class Person < ApplicationResource
  schema do
    integer 'age'
  end
end

p = Person.new
p.age = "18"
p.age # => 18

To set all resources application-wide to cast values, set config.active_resource.cast_values:

  # config/application.rb
  config.active_resource.cast_values = true

@seanpdoyle seanpdoyle force-pushed the attributes-without-casting branch 2 times, most recently from d1d2bca to d789946 Compare November 27, 2024 14:24
@jlurena
Copy link

jlurena commented Nov 29, 2024

Is this a draft? Once you are done maybe we can combine both our feature enhancements (mine is here: #409) and create a new major version of this gem.

@rails @byroot thoughts?

@seanpdoyle seanpdoyle force-pushed the attributes-without-casting branch 5 times, most recently from 72a380b to 408844c Compare January 25, 2025 20:08
The `schema { ... }` interface pre-dates the Active Model Attributes API
(defined as early as [v5.2.0][]), but clearly draws inspiration from
Active Record's Database Schema and Attribute casting (which was
extracted into `ActiveModel::Attributes`).

However, the type information captured in `schema { ... }` blocks or
assigned as `Hash` arguments to `schema=` is purely inert metadata.

Proposal
---

This commit aims to integrate with [ActiveModel::Model][] and
[ActiveModel::Attributes][]. Through the introduction of both modules,
subclasses of `ActiveResource::Schema` can benefit from type casting
attributes and constructing instances with default values.

This commit makes minimally incremental changes, prioritizing backwards
compatibility. The reliance on `#respond_to_missing?` and
`#method_missing` is left largely unchanged. Similarly, the `Schema`
interface continues to provide metadata about its attributes through the
`Schema#attr` method (instead of reading from
`ActiveModel::Attributes#attribute_names` or
`ActiveModel::Attributes.attribute_types`).

API Changes
---

To cast values to their specified types, declare the Schema with the
`:cast_values` set to true.

```ruby
class Person < ActiveResource::Base
  schema cast_values: true do
    integer 'age'
  end
end

p = Person.new
p.age = "18"
p.age # => 18
```

To configure inheriting resources to cast values, set the `cast_values`
class attribute:

```ruby
class ApplicationResource < ActiveResource::Base
  self.cast_values = true
end

class Person < ApplicationResource
  schema do
    integer 'age'
  end
end

p = Person.new
p.age = "18"
p.age # => 18
```

To set all resources application-wide to cast values, set
`config.active_resource.cast_values`:

```ruby
  # config/application.rb
  config.active_resource.cast_values = true
```

[v5.2.0]: https://api.rubyonrails.org/v5.2.0/classes/ActiveModel/Attributes/ClassMethods.html
[ActiveModel::Model]: https://api.rubyonrails.org/classes/ActiveModel/Model.html
[ActiveModel::Attributes]: https://api.rubyonrails.org/classes/ActiveModel/Attributes/ClassMethods.html
@seanpdoyle seanpdoyle force-pushed the attributes-without-casting branch from 408844c to 5e5344a Compare January 27, 2025 03:27
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

Successfully merging this pull request may close these issues.

2 participants