| 
									
										
										
										
											2018-05-11 04:49:12 -05:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Web::PushNotificationWorker | 
					
						
							|  |  |  |   include Sidekiq::Worker | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  |   sidekiq_options queue: 'push', retry: 5
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   TTL     = 48.hours.to_s | 
					
						
							|  |  |  |   URGENCY = 'normal' | 
					
						
							| 
									
										
										
										
											2018-05-11 04:49:12 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   def perform(subscription_id, notification_id) | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  |     @subscription = Web::PushSubscription.find(subscription_id) | 
					
						
							|  |  |  |     @notification = Notification.find(notification_id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Polymorphically associated activity could have been deleted | 
					
						
							|  |  |  |     # in the meantime, so we have to double-check before proceeding | 
					
						
							|  |  |  |     return unless @notification.activity.present? && @subscription.pushable?(@notification) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     payload = @subscription.encrypt(push_notification_json) | 
					
						
							| 
									
										
										
										
											2018-05-11 04:49:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  |     request_pool.with(@subscription.audience) do |http_client| | 
					
						
							|  |  |  |       request = Request.new(:post, @subscription.endpoint, body: payload.fetch(:ciphertext), http_client: http_client) | 
					
						
							| 
									
										
										
										
											2019-09-13 12:15:47 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  |       request.add_headers( | 
					
						
							| 
									
										
										
										
											2023-02-19 23:58:28 -06:00
										 |  |  |         'Content-Type' => 'application/octet-stream', | 
					
						
							|  |  |  |         'Ttl' => TTL, | 
					
						
							|  |  |  |         'Urgency' => URGENCY, | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  |         'Content-Encoding' => 'aesgcm', | 
					
						
							| 
									
										
										
										
											2023-02-19 23:58:28 -06:00
										 |  |  |         'Encryption' => "salt=#{Webpush.encode64(payload.fetch(:salt)).delete('=')}", | 
					
						
							|  |  |  |         'Crypto-Key' => "dh=#{Webpush.encode64(payload.fetch(:server_public_key)).delete('=')};#{@subscription.crypto_key_header}", | 
					
						
							|  |  |  |         'Authorization' => @subscription.authorization_header | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  |       ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       request.perform do |response| | 
					
						
							|  |  |  |         # If the server responds with an error in the 4xx range | 
					
						
							|  |  |  |         # that isn't about rate-limiting or timeouts, we can | 
					
						
							|  |  |  |         # assume that the subscription is invalid or expired | 
					
						
							|  |  |  |         # and must be removed | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (400..499).cover?(response.code) && ![408, 429].include?(response.code) | 
					
						
							|  |  |  |           @subscription.destroy! | 
					
						
							|  |  |  |         elsif !(200...300).cover?(response.code) | 
					
						
							|  |  |  |           raise Mastodon::UnexpectedResponseError, response | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2019-09-13 12:15:47 -05:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2018-05-11 04:49:12 -05:00
										 |  |  |   rescue ActiveRecord::RecordNotFound | 
					
						
							|  |  |  |     true | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   private | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def push_notification_json | 
					
						
							| 
									
										
										
										
											2022-05-28 07:32:08 -05:00
										 |  |  |     json = I18n.with_locale(@subscription.locale.presence || I18n.default_locale) do | 
					
						
							| 
									
										
										
										
											2021-04-12 07:25:34 -05:00
										 |  |  |       ActiveModelSerializers::SerializableResource.new( | 
					
						
							|  |  |  |         @notification, | 
					
						
							|  |  |  |         serializer: Web::NotificationSerializer, | 
					
						
							|  |  |  |         scope: @subscription, | 
					
						
							|  |  |  |         scope_name: :current_push_subscription | 
					
						
							|  |  |  |       ).as_json | 
					
						
							|  |  |  |     end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Oj.dump(json) | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def request_pool | 
					
						
							|  |  |  |     RequestPool.current | 
					
						
							|  |  |  |   end | 
					
						
							| 
									
										
										
										
											2018-05-11 04:49:12 -05:00
										 |  |  | end |