mirror of
https://github.com/openfoodfoundation/openfoodnetwork
synced 2026-01-27 21:06:49 +00:00
57 lines
1.6 KiB
Ruby
57 lines
1.6 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
# Request a JSON document from a DFC API with authentication.
|
|
#
|
|
# All DFC API interactions are authenticated via OIDC tokens. If the user's
|
|
# access token is expired, we try to get a new one with the user's refresh
|
|
# token.
|
|
class DfcRequest
|
|
def initialize(user)
|
|
@user = user
|
|
end
|
|
|
|
def call(url, data = nil, method: nil)
|
|
begin
|
|
request = OidcRequest.new(@user.oidc_account.token)
|
|
response = request.call(url, data, method:)
|
|
rescue Faraday::UnauthorizedError, Faraday::ForbiddenError
|
|
raise unless token_stale?
|
|
|
|
# If access was denied and our token is stale then refresh and retry:
|
|
refresh_access_token!
|
|
request = OidcRequest.new(@user.oidc_account.token)
|
|
response = request.call(url, data, method:)
|
|
end
|
|
|
|
response.body
|
|
end
|
|
|
|
private
|
|
|
|
def token_stale?
|
|
@user.oidc_account.updated_at < 15.minutes.ago
|
|
end
|
|
|
|
def refresh_access_token!
|
|
strategy = OmniAuth::Strategies::OpenIDConnect.new(
|
|
Rails.application,
|
|
Devise.omniauth_configs[:openid_connect].options
|
|
# Don't try to call `Devise.omniauth(:openid_connect)` first.
|
|
# It results in an empty config hash and we lose our config.
|
|
)
|
|
client = strategy.client
|
|
client.token_endpoint = strategy.config.token_endpoint
|
|
client.refresh_token = @user.oidc_account.refresh_token
|
|
|
|
token = client.access_token!
|
|
|
|
@user.oidc_account.update!(
|
|
token: token.access_token,
|
|
refresh_token: token.refresh_token
|
|
)
|
|
rescue Rack::OAuth2::Client::Error
|
|
@user.oidc_account.update!(token: nil, refresh_token: nil)
|
|
raise
|
|
end
|
|
end
|