mirror of
https://github.com/openfoodfoundation/openfoodnetwork
synced 2026-02-05 22:26:07 +00:00
Access tokens are only valid for half an hour. So if requesting a DFC API fails, it's likely due to an expired token and we refresh it.
70 lines
1.7 KiB
Ruby
70 lines
1.7 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require "private_address_check"
|
|
require "private_address_check/tcpsocket_ext"
|
|
|
|
# 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 get(url)
|
|
response = request(url)
|
|
|
|
return response.body if response.status == 200
|
|
|
|
return "" if @user.oidc_account.updated_at > 15.minutes.ago
|
|
|
|
refresh_access_token!
|
|
|
|
response = request(url)
|
|
response.body
|
|
end
|
|
|
|
private
|
|
|
|
def request(url)
|
|
connection = Faraday.new(
|
|
request: { timeout: 30 },
|
|
headers: {
|
|
'Content-Type' => 'application/json',
|
|
'Authorization' => "Bearer #{@user.oidc_account.token}",
|
|
}
|
|
)
|
|
|
|
only_public_connections do
|
|
connection.get(url)
|
|
end
|
|
end
|
|
|
|
def only_public_connections(&)
|
|
return yield if Rails.env.development?
|
|
|
|
PrivateAddressCheck.only_public_connections(&)
|
|
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
|
|
)
|
|
end
|
|
end
|