Skip to content

Commit 0c728a5

Browse files
committed
feat: add global register
1 parent 83887e8 commit 0c728a5

12 files changed

+323
-40
lines changed

.rubocop_todo.yml

+35-3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,17 @@
11
# This configuration was generated by
22
# `rubocop --auto-gen-config`
3-
# on 2025-03-12 04:30:42 UTC using RuboCop version 1.73.2.
3+
# on 2025-03-12 09:47:59 UTC using RuboCop version 1.73.2.
44
# The point is for the user to remove these configuration records
55
# one by one as the offenses are removed from the code base.
66
# Note that changes in the inspected code, or installation of new
77
# versions of RuboCop, may require this file to be generated again.
88

9+
# Offense count: 1
10+
# This cop supports safe autocorrection (--autocorrect).
11+
Layout/EmptyLineAfterMagicComment:
12+
Exclude:
13+
- 'lib/lutaml/hal/global_register.rb'
14+
915
# Offense count: 3
1016
# Configuration parameters: AllowedMethods, AllowedPatterns, CountRepeatedAttributes.
1117
Metrics/AbcSize:
@@ -15,9 +21,14 @@ Metrics/AbcSize:
1521
# Configuration parameters: CountComments, CountAsOne, AllowedMethods, AllowedPatterns.
1622
# AllowedMethods: refine
1723
Metrics/BlockLength:
18-
Max: 177
24+
Max: 188
1925

20-
# Offense count: 8
26+
# Offense count: 1
27+
# Configuration parameters: AllowedMethods, AllowedPatterns.
28+
Metrics/CyclomaticComplexity:
29+
Max: 9
30+
31+
# Offense count: 9
2132
# Configuration parameters: CountComments, CountAsOne, AllowedMethods, AllowedPatterns.
2233
Metrics/MethodLength:
2334
Max: 21
@@ -27,6 +38,11 @@ Metrics/MethodLength:
2738
Metrics/ParameterLists:
2839
Max: 7
2940

41+
# Offense count: 1
42+
# Configuration parameters: AllowedMethods, AllowedPatterns.
43+
Metrics/PerceivedComplexity:
44+
Max: 9
45+
3046
# Offense count: 1
3147
Naming/AccessorMethodName:
3248
Exclude:
@@ -55,3 +71,19 @@ Naming/HeredocDelimiterNaming:
5571
Naming/VariableNumber:
5672
Exclude:
5773
- 'spec/lutaml/hal/page_spec.rb'
74+
75+
# Offense count: 1
76+
# This cop supports unsafe autocorrection (--autocorrect-all).
77+
# Configuration parameters: EnforcedStyle.
78+
# SupportedStyles: always, always_true, never
79+
Style/FrozenStringLiteralComment:
80+
Exclude:
81+
- '**/*.arb'
82+
- 'lib/lutaml/hal/global_register.rb'
83+
84+
# Offense count: 1
85+
# This cop supports unsafe autocorrection (--autocorrect-all).
86+
# Configuration parameters: Mode.
87+
Style/StringConcatenation:
88+
Exclude:
89+
- 'lib/lutaml/hal/model_register.rb'

README.adoc

+82-4
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,10 @@ A registry for managing HAL resource models and their endpoints. It allows you
7171
to register models, define their relationships, and fetch resources from the
7272
API.
7373

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+
7478
`Lutaml::Hal::Resource`::
7579
A base class for defining HAL resource models. It includes methods for
7680
defining attributes, links, and key-value mappings for resources.
@@ -96,6 +100,8 @@ At the data definition phase:
96100
. Define the API endpoint using the `Client` class.
97101
. Create a `ModelRegister` to manage the resource models and their
98102
respective endpoints.
103+
. (optional) Create a `GlobalRegister` to manage one or more `ModelRegister`
104+
instances. It is necessary for automatic Link resolution.
99105
. Define the resource models using the `Resource` class.
100106
. Register the models with the `ModelRegister` and define their
101107
relationships using the `add_endpoint` method.
@@ -145,10 +151,38 @@ require 'lutaml-hal'
145151
146152
# Create a new client with API endpoint
147153
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)
149155
# Or set client later, `register.client = client`
150156
----
151157

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+
152186

153187
=== Defining HAL resource models
154188

@@ -158,7 +192,7 @@ A HAL resource is defined by creating a subclass of the `Resource` class and
158192
defining its attributes, links, and key-value mappings.
159193

160194
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
162196
modelling and serialization capabilities.
163197

164198
The declaration of attributes, links, and key-value mappings for a HAL resource
@@ -780,16 +814,34 @@ product_index
780814
====
781815

782816

817+
[[fetching_resource_via_link_realization]]
783818
=== Fetching a resource via link realization
784819

785820
Given a resource index that contains links to resources, the individual resource
786821
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.
788823

789824
Given a `Link` object, the `realize` method fetches the resource from the API
790825
using the provided `register`.
791826

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:
793845

794846
[source,ruby]
795847
----
@@ -813,12 +865,26 @@ NOTE: It is possible to use the `realize` method on a link object using another
813865
`ModelRegister` instance. This is useful when you want to resolve a link
814866
using a different API endpoint or a different set of resource models.
815867

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+
816877
.Dynamically realizing a resource from the collection using links
817878
[example]
818879
====
819880
[source,ruby]
820881
----
882+
# Without a GlobalRegister
821883
product_2 = product_index.links.products.last.realize(register)
884+
885+
# With a GlobalRegister
886+
product_2 = product_index.links.products.last.realize
887+
822888
# => client.get('/products/2')
823889
# => {
824890
# "id": 2,
@@ -842,9 +908,16 @@ product_2
842908
# <ProductLink href: "/products/4", title: "Product 4">,
843909
# <ProductLink href: "/products/6", title: "Product 6">
844910
# ]}>
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
845917
----
846918
====
847919

920+
848921
=== Handling HAL pages / pagination
849922

850923
The `Lutaml::Hal::Page` class is used to handle pagination in HAL APIs.
@@ -907,7 +980,12 @@ page_1
907980
# next: #<ResourceIndexLink href: "/resources?page=2&items=10">,
908981
# last: #<ResourceIndexLink href: "/resources?page=10&items=10">>>
909982
983+
# Without a GlobalRegister
910984
page_2 = page.links.next.realize(register)
985+
986+
# With a GlobalRegister
987+
page_2 = page.links.next.realize
988+
911989
# => client.get('/resources?page=2&items=10')
912990
# => #<ResourceIndex page: 2, pages: 10, limit: 10, total: 100,
913991
# links: #<ResourceIndexLinks

lib/lutaml/hal.rb

+7
Original file line numberDiff line numberDiff line change
@@ -5,13 +5,20 @@
55
module Lutaml
66
# HAL implementation for Lutaml
77
module Hal
8+
REGISTER_ID_ATTR_NAME = '_global_register_id'
9+
10+
def self.debug_log(message)
11+
puts "[Lutaml::Hal] DEBUG: #{message}" if ENV['DEBUG_API']
12+
end
813
end
914
end
1015

1116
require_relative 'hal/version'
1217
require_relative 'hal/errors'
1318
require_relative 'hal/link'
19+
require_relative 'hal/link_set'
1420
require_relative 'hal/resource'
1521
require_relative 'hal/page'
22+
require_relative 'hal/global_register'
1623
require_relative 'hal/model_register'
1724
require_relative 'hal/client'

lib/lutaml/hal/client.rb

+12-5
Original file line numberDiff line numberDiff line change
@@ -19,12 +19,19 @@ def initialize(options = {})
1919
@debug = options[:debug] || !ENV['DEBUG_API'].nil?
2020
@cache = options[:cache] || {}
2121
@cache_enabled = options[:cache_enabled] || false
22+
23+
@api_url = strip_api_url(@api_url)
24+
end
25+
26+
# Strip any trailing slash from the API URL
27+
def strip_api_url(url)
28+
url.sub(%r{/\Z}, '')
2229
end
2330

2431
# Get a resource by its full URL
2532
def get_by_url(url, params = {})
2633
# Strip API endpoint if it's included
27-
path = url.sub(%r{^#{@api_url}/}, '')
34+
path = strip_api_url(url)
2835
get(path, params)
2936
end
3037

@@ -62,7 +69,7 @@ def create_connection
6269
end
6370

6471
def handle_response(response, url)
65-
debug_log(response, url) if @debug
72+
debug_api_log(response, url) if @debug
6673

6774
case response.status
6875
when 200..299
@@ -80,11 +87,11 @@ def handle_response(response, url)
8087
end
8188
end
8289

83-
def debug_log(response, url)
90+
def debug_api_log(response, url)
8491
if defined?(Rainbow)
85-
puts Rainbow("\n===== DEBUG: HAL API REQUEST =====").blue
92+
puts Rainbow("\n===== Lutaml::Hal DEBUG: HAL API REQUEST =====").blue
8693
else
87-
puts "\n===== DEBUG: HAL API REQUEST ====="
94+
puts "\n===== Lutaml::Hal DEBUG: HAL API REQUEST ====="
8895
end
8996

9097
puts "URL: #{url}"

lib/lutaml/hal/global_register.rb

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
# frozen_string_literal: true
2+
3+
require 'singleton'
4+
5+
module Lutaml
6+
module Hal
7+
# Global register for model registers
8+
# This class is a singleton that manages the registration and retrieval of model registers.
9+
# It ensures that each model register is unique and provides a way to access them globally.
10+
#
11+
# @example
12+
# global_register = GlobalRegister.instance
13+
# global_register.register(:example, ExampleModelRegister.new)
14+
# example_register = global_register.get(:example)
15+
class GlobalRegister
16+
include Singleton
17+
18+
def initialize
19+
@model_registers = {}
20+
end
21+
22+
def register(name, model_register)
23+
if @model_registers[name] && @model_registers[name] != model_register
24+
raise "Model register with name #{name} replacing another one" \
25+
" (#{@model_registers[name].inspect} vs #{model_register.inspect})"
26+
end
27+
28+
@model_registers[name] = model_register
29+
end
30+
31+
def get(name)
32+
raise "Model register with name #{name} not found" unless @model_registers[name]
33+
34+
@model_registers[name]
35+
end
36+
37+
def delete(name)
38+
return unless @model_registers[name]
39+
40+
@model_registers.delete(name)
41+
end
42+
end
43+
end
44+
end

lib/lutaml/hal/link.rb

+31-3
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,10 @@ module Lutaml
77
module Hal
88
# HAL Link representation with realization capability
99
class Link < Lutaml::Model::Serializable
10+
# This is the model register that has fetched the origin of this link, and
11+
# will be used to resolve unless overriden in resource#realize()
12+
attr_accessor Hal::REGISTER_ID_ATTR_NAME.to_sym
13+
1014
attribute :href, :string
1115
attribute :title, :string
1216
attribute :name, :string
@@ -16,9 +20,33 @@ class Link < Lutaml::Model::Serializable
1620
attribute :profile, :string
1721
attribute :lang, :string
1822

19-
# Fetch the actual resource this link points to
20-
def realize(register)
21-
register.resolve_and_cast(href)
23+
# Fetch the actual resource this link points to.
24+
# This method will use the global register according to the source of the Link object.
25+
# If the Link does not have a register, a register needs to be provided explicitly
26+
# via the `register:` parameter.
27+
def realize(register: nil)
28+
register = find_register(register)
29+
raise "No register provided for link resolution (class: #{self.class}, href: #{href})" if register.nil?
30+
31+
Hal.debug_log "Resolving link href: #{href} using register"
32+
register.resolve_and_cast(self, href)
33+
end
34+
35+
private
36+
37+
def find_register(explicit_register)
38+
return explicit_register if explicit_register
39+
40+
register_id = instance_variable_get("@#{Hal::REGISTER_ID_ATTR_NAME}")
41+
return nil if register_id.nil?
42+
43+
register = Lutaml::Hal::GlobalRegister.instance.get(register_id)
44+
if register.nil?
45+
raise 'GlobalRegister in use but unable to find the register. '\
46+
'Please provide a register to the `#realize` method to resolve the link'
47+
end
48+
49+
register
2250
end
2351
end
2452
end

0 commit comments

Comments
 (0)