2024-10-02 07:11:52 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
class WebPushRequest
|
|
|
|
SIGNATURE_ALGORITHM = 'p256ecdsa'
|
2025-01-14 10:14:00 +01:00
|
|
|
LEGACY_AUTH_HEADER = 'WebPush'
|
|
|
|
STANDARD_AUTH_HEADER = 'vapid'
|
2024-10-02 07:11:52 -04:00
|
|
|
PAYLOAD_EXPIRATION = 24.hours
|
|
|
|
JWT_ALGORITHM = 'ES256'
|
|
|
|
JWT_TYPE = 'JWT'
|
|
|
|
|
|
|
|
attr_reader :web_push_subscription
|
|
|
|
|
|
|
|
delegate(
|
2025-01-14 10:14:00 +01:00
|
|
|
:standard,
|
2024-10-02 07:11:52 -04:00
|
|
|
:endpoint,
|
|
|
|
:key_auth,
|
|
|
|
:key_p256dh,
|
|
|
|
to: :web_push_subscription
|
|
|
|
)
|
|
|
|
|
|
|
|
def initialize(web_push_subscription)
|
|
|
|
@web_push_subscription = web_push_subscription
|
|
|
|
end
|
|
|
|
|
|
|
|
def audience
|
|
|
|
@audience ||= Addressable::URI.parse(endpoint).normalized_site
|
|
|
|
end
|
|
|
|
|
2025-01-14 10:14:00 +01:00
|
|
|
def legacy_authorization_header
|
|
|
|
[LEGACY_AUTH_HEADER, encoded_json_web_token].join(' ')
|
2024-10-02 07:11:52 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def crypto_key_header
|
|
|
|
[SIGNATURE_ALGORITHM, vapid_key.public_key_for_push_header].join('=')
|
|
|
|
end
|
|
|
|
|
2025-01-14 10:14:00 +01:00
|
|
|
def legacy_encrypt(payload)
|
2025-01-13 13:46:32 +01:00
|
|
|
Webpush::Legacy::Encryption.encrypt(payload, key_p256dh, key_auth)
|
2024-10-02 07:11:52 -04:00
|
|
|
end
|
|
|
|
|
2025-01-14 10:14:00 +01:00
|
|
|
def standard_authorization_header
|
|
|
|
[STANDARD_AUTH_HEADER, standard_vapid_value].join(' ')
|
|
|
|
end
|
|
|
|
|
|
|
|
def standard_encrypt(payload)
|
|
|
|
Webpush::Encryption.encrypt(payload, key_p256dh, key_auth)
|
|
|
|
end
|
|
|
|
|
|
|
|
def legacy
|
|
|
|
!standard
|
|
|
|
end
|
|
|
|
|
2024-10-02 07:11:52 -04:00
|
|
|
private
|
|
|
|
|
2025-01-14 10:14:00 +01:00
|
|
|
def standard_vapid_value
|
|
|
|
"t=#{encoded_json_web_token},k=#{vapid_key.public_key_for_push_header}"
|
|
|
|
end
|
|
|
|
|
2024-10-02 07:11:52 -04:00
|
|
|
def encoded_json_web_token
|
|
|
|
JWT.encode(
|
|
|
|
web_token_payload,
|
|
|
|
vapid_key.curve,
|
|
|
|
JWT_ALGORITHM,
|
|
|
|
typ: JWT_TYPE
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def web_token_payload
|
|
|
|
{
|
|
|
|
aud: audience,
|
|
|
|
exp: PAYLOAD_EXPIRATION.from_now.to_i,
|
|
|
|
sub: payload_subject,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
def payload_subject
|
|
|
|
[:mailto, contact_email].join(':')
|
|
|
|
end
|
|
|
|
|
|
|
|
def vapid_key
|
|
|
|
@vapid_key ||= Webpush::VapidKey.from_keys(
|
|
|
|
Rails.configuration.x.vapid_public_key,
|
|
|
|
Rails.configuration.x.vapid_private_key
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def contact_email
|
|
|
|
@contact_email ||= ::Setting.site_contact_email
|
|
|
|
end
|
|
|
|
end
|