@@ -71,6 +71,10 @@ A registry for managing HAL resource models and their endpoints. It allows you
71
71
to register models, define their relationships, and fetch resources from the
72
72
API.
73
73
74
+ `Lutaml::Hal::GlobalRegister`::
75
+ A global registry (Singleton) for managing ModelRegisters and facilitating model
76
+ resolution across different resources. Its usage is optional.
77
+
74
78
`Lutaml::Hal::Resource`::
75
79
A base class for defining HAL resource models. It includes methods for
76
80
defining attributes, links, and key-value mappings for resources.
@@ -96,6 +100,8 @@ At the data definition phase:
96
100
. Define the API endpoint using the `Client` class.
97
101
. Create a `ModelRegister` to manage the resource models and their
98
102
respective endpoints.
103
+ . (optional) Create a `GlobalRegister` to manage one or more `ModelRegister`
104
+ instances. It is necessary for automatic Link resolution.
99
105
. Define the resource models using the `Resource` class.
100
106
. Register the models with the `ModelRegister` and define their
101
107
relationships using the `add_endpoint` method.
@@ -145,10 +151,38 @@ require 'lutaml-hal'
145
151
146
152
# Create a new client with API endpoint
147
153
client = Lutaml::Hal::Client.new(api_url: 'https://api.example.com')
148
- register = Lutaml::Hal::ModelRegister.new(client: client)
154
+ register = Lutaml::Hal::ModelRegister.new(name: :my_model_register, client: client)
149
155
# Or set client later, `register.client = client`
150
156
----
151
157
158
+ The `name:` parameter is used to identify the `ModelRegister` instance.
159
+
160
+ === Creating a HAL global register
161
+
162
+ The `GlobalRegister` class is a singleton that manages one or more
163
+ `ModelRegister` instances.
164
+
165
+ It is optional, but is required for automatic realization of models from Link
166
+ objects. See <<fetching_resource_via_link_realization>> for more details.
167
+
168
+ [source,ruby]
169
+ ----
170
+ require 'lutaml-hal'
171
+
172
+ # Create a new client with API endpoint
173
+ client = Lutaml::Hal::Client.new(api_url: 'https://api.example.com')
174
+ register = Lutaml::Hal::ModelRegister.new(name: :my_model_register, client: client)
175
+
176
+ # Register the ModelRegister with the global register
177
+ global_register = Lutaml::Hal::GlobalRegister.instance.register(:my_model_register, register)
178
+
179
+ # Obtain the global register
180
+ global_register.get(:my_model_register)
181
+
182
+ # Delete a register mapping
183
+ global_register.delete(:my_model_register)
184
+ ----
185
+
152
186
153
187
=== Defining HAL resource models
154
188
@@ -158,7 +192,7 @@ A HAL resource is defined by creating a subclass of the `Resource` class and
158
192
defining its attributes, links, and key-value mappings.
159
193
160
194
The `Resource` class is the base class for defining HAL resource models.
161
- It inherits from `Lutaml::Model::Serialization `, which provides data
195
+ It inherits from `Lutaml::Model::Serializable `, which provides data
162
196
modelling and serialization capabilities.
163
197
164
198
The declaration of attributes, links, and key-value mappings for a HAL resource
@@ -780,16 +814,34 @@ product_index
780
814
====
781
815
782
816
817
+ [[fetching_resource_via_link_realization]]
783
818
=== Fetching a resource via link realization
784
819
785
820
Given a resource index that contains links to resources, the individual resource
786
821
links can be "realized" as actual model instances through the
787
- `Link#realize(register)` method which dynamically retrieves the resource.
822
+ `Link#realize(register: )` method which dynamically retrieves the resource.
788
823
789
824
Given a `Link` object, the `realize` method fetches the resource from the API
790
825
using the provided `register`.
791
826
792
- Syntax:
827
+ There are two ways a resource gets realized from a `Link` object:
828
+
829
+ * If a `Lutaml::Hal::GlobalRegister` is used, and the `Link` object originated
830
+ from a fetch using a `ModelRegister` then the `realize` method has sufficient
831
+ information to automatically fetch the resource from the API using the same
832
+ `register`.
833
+ +
834
+ NOTE: This relies on the `Hal::REGISTER_ID_ATTR_NAME` attribute to be set
835
+ in the `ModelRegister` class. This attribute is used to identify the
836
+ resource endpoint ID in the URL.
837
+
838
+ * If a `GlobalRegister` is not used, even if the Link object originated
839
+ from a fetch using a `ModelRegister`, the `realize` method does not have sufficient
840
+ information to fetch the resource from the API using the same
841
+ `register`. In this case an explicit `register` must be provided to the
842
+ `realize(register: ...)` method.
843
+
844
+ Syntax for standalone usage:
793
845
794
846
[source,ruby]
795
847
----
@@ -813,12 +865,26 @@ NOTE: It is possible to use the `realize` method on a link object using another
813
865
`ModelRegister` instance. This is useful when you want to resolve a link
814
866
using a different API endpoint or a different set of resource models.
815
867
868
+ Syntax when using a `GlobalRegister`:
869
+
870
+ [source,ruby]
871
+ ----
872
+ resource_index = model_register.fetch(:resource_index)
873
+ resource_index.links.products.first.realize
874
+ # => client.get('/resources/1')
875
+ ----
876
+
816
877
.Dynamically realizing a resource from the collection using links
817
878
[example]
818
879
====
819
880
[source,ruby]
820
881
----
882
+ # Without a GlobalRegister
821
883
product_2 = product_index.links.products.last.realize(register)
884
+
885
+ # With a GlobalRegister
886
+ product_2 = product_index.links.products.last.realize
887
+
822
888
# => client.get('/products/2')
823
889
# => {
824
890
# "id": 2,
@@ -842,9 +908,16 @@ product_2
842
908
# <ProductLink href: "/products/4", title: "Product 4">,
843
909
# <ProductLink href: "/products/6", title: "Product 6">
844
910
# ]}>
911
+
912
+ # Without a GlobalRegister
913
+ product_2_related_1 = product_2.links.related.first.realize(register)
914
+
915
+ # With a GlobalRegister
916
+ product_2_related_1 = product_2.links.related.first.realize
845
917
----
846
918
====
847
919
920
+
848
921
=== Handling HAL pages / pagination
849
922
850
923
The `Lutaml::Hal::Page` class is used to handle pagination in HAL APIs.
@@ -907,7 +980,12 @@ page_1
907
980
# next: #<ResourceIndexLink href: "/resources?page=2&items=10">,
908
981
# last: #<ResourceIndexLink href: "/resources?page=10&items=10">>>
909
982
983
+ # Without a GlobalRegister
910
984
page_2 = page.links.next.realize(register)
985
+
986
+ # With a GlobalRegister
987
+ page_2 = page.links.next.realize
988
+
911
989
# => client.get('/resources?page=2&items=10')
912
990
# => #<ResourceIndex page: 2, pages: 10, limit: 10, total: 100,
913
991
# links: #<ResourceIndexLinks
0 commit comments