Skip to content
This repository has been archived by the owner on Dec 21, 2023. It is now read-only.

Commit

Permalink
Fix streaming API allowing connections to persist after access token …
Browse files Browse the repository at this point in the history
…invalidation (mastodon#15111)

Fix mastodon#14816
  • Loading branch information
Gargron authored Nov 12, 2020
1 parent 8532429 commit aa10200
Show file tree
Hide file tree
Showing 4 changed files with 109 additions and 7 deletions.
17 changes: 17 additions & 0 deletions app/lib/access_token_extension.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
# frozen_string_literal: true

module AccessTokenExtension
extend ActiveSupport::Concern

included do
after_commit :push_to_streaming_api
end

def revoke(clock = Time)
update(revoked_at: clock.now.utc)
end

def push_to_streaming_api
Redis.current.publish("timeline:access_token:#{id}", Oj.dump(event: :kill)) if revoked? || destroyed?
end
end
16 changes: 10 additions & 6 deletions app/models/session_activation.rb
Original file line number Diff line number Diff line change
Expand Up @@ -70,12 +70,16 @@ def assign_user_agent
end

def assign_access_token
superapp = Doorkeeper::Application.find_by(superapp: true)
self.access_token = Doorkeeper::AccessToken.create!(access_token_attributes)
end

self.access_token = Doorkeeper::AccessToken.create!(application_id: superapp&.id,
resource_owner_id: user_id,
scopes: 'read write follow',
expires_in: Doorkeeper.configuration.access_token_expires_in,
use_refresh_token: Doorkeeper.configuration.refresh_token_enabled?)
def access_token_attributes
{
application_id: Doorkeeper::Application.find_by(superapp: true)&.id,
resource_owner_id: user_id,
scopes: 'read write follow',
expires_in: Doorkeeper.configuration.access_token_expires_in,
use_refresh_token: Doorkeeper.configuration.refresh_token_enabled?,
}
end
end
1 change: 1 addition & 0 deletions config/application.rb
Original file line number Diff line number Diff line change
Expand Up @@ -140,6 +140,7 @@ class Application < Rails::Application
Doorkeeper::AuthorizationsController.layout 'modal'
Doorkeeper::AuthorizedApplicationsController.layout 'admin'
Doorkeeper::Application.send :include, ApplicationExtension
Doorkeeper::AccessToken.send :include, AccessTokenExtension
Devise::FailureApp.send :include, AbstractController::Callbacks
Devise::FailureApp.send :include, HttpAcceptLanguage::EasyAccess
Devise::FailureApp.send :include, Localized
Expand Down
82 changes: 81 additions & 1 deletion streaming/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -294,7 +294,7 @@ const startWorker = (workerId) => {
return;
}

client.query('SELECT oauth_access_tokens.resource_owner_id, users.account_id, users.chosen_languages, oauth_access_tokens.scopes, devices.device_id FROM oauth_access_tokens INNER JOIN users ON oauth_access_tokens.resource_owner_id = users.id LEFT OUTER JOIN devices ON oauth_access_tokens.id = devices.access_token_id WHERE oauth_access_tokens.token = $1 AND oauth_access_tokens.revoked_at IS NULL LIMIT 1', [token], (err, result) => {
client.query('SELECT oauth_access_tokens.id, oauth_access_tokens.resource_owner_id, users.account_id, users.chosen_languages, oauth_access_tokens.scopes, devices.device_id FROM oauth_access_tokens INNER JOIN users ON oauth_access_tokens.resource_owner_id = users.id LEFT OUTER JOIN devices ON oauth_access_tokens.id = devices.access_token_id WHERE oauth_access_tokens.token = $1 AND oauth_access_tokens.revoked_at IS NULL LIMIT 1', [token], (err, result) => {
done();

if (err) {
Expand All @@ -310,6 +310,7 @@ const startWorker = (workerId) => {
return;
}

req.accessTokenId = result.rows[0].id;
req.scopes = result.rows[0].scopes.split(' ');
req.accountId = result.rows[0].account_id;
req.chosenLanguages = result.rows[0].chosen_languages;
Expand Down Expand Up @@ -450,6 +451,55 @@ const startWorker = (workerId) => {
});
};

/**
* @typedef SystemMessageHandlers
* @property {function(): void} onKill
*/

/**
* @param {any} req
* @param {SystemMessageHandlers} eventHandlers
* @return {function(string): void}
*/
const createSystemMessageListener = (req, eventHandlers) => {
return message => {
const json = parseJSON(message);

if (!json) return;

const { event } = json;

log.silly(req.requestId, `System message for ${req.accountId}: ${event}`);

if (event === 'kill') {
log.verbose(req.requestId, `Closing connection for ${req.accountId} due to expired access token`);
eventHandlers.onKill();
}
}
};

/**
* @param {any} req
* @param {any} res
*/
const subscribeHttpToSystemChannel = (req, res) => {
const systemChannelId = `timeline:access_token:${req.accessTokenId}`;

const listener = createSystemMessageListener(req, {

onKill () {
res.end();
},

});

res.on('close', () => {
unsubscribe(`${redisPrefix}${systemChannelId}`, listener);
});

subscribe(`${redisPrefix}${systemChannelId}`, listener);
};

/**
* @param {any} req
* @param {any} res
Expand All @@ -462,6 +512,8 @@ const startWorker = (workerId) => {
}

accountFromRequest(req, alwaysRequireAuth).then(() => checkScopes(req, channelNameFromPath(req))).then(() => {
subscribeHttpToSystemChannel(req, res);
}).then(() => {
next();
}).catch(err => {
next(err);
Expand Down Expand Up @@ -536,7 +588,9 @@ const startWorker = (workerId) => {

const listener = message => {
const json = parseJSON(message);

if (!json) return;

const { event, payload, queued_at } = json;

const transmit = () => {
Expand Down Expand Up @@ -902,6 +956,28 @@ const startWorker = (workerId) => {
socket.send(JSON.stringify({ error: err.toString() }));
});

/**
* @param {WebSocketSession} session
*/
const subscribeWebsocketToSystemChannel = ({ socket, request, subscriptions }) => {
const systemChannelId = `timeline:access_token:${request.accessTokenId}`;

const listener = createSystemMessageListener(request, {

onKill () {
socket.close();
},

});

subscribe(`${redisPrefix}${systemChannelId}`, listener);

subscriptions[systemChannelId] = {
listener,
stopHeartbeat: () => {},
};
};

/**
* @param {string|string[]} arrayOrString
* @return {string}
Expand Down Expand Up @@ -948,7 +1024,9 @@ const startWorker = (workerId) => {

ws.on('message', data => {
const json = parseJSON(data);

if (!json) return;

const { type, stream, ...params } = json;

if (type === 'subscribe') {
Expand All @@ -960,6 +1038,8 @@ const startWorker = (workerId) => {
}
});

subscribeWebsocketToSystemChannel(session);

if (location.query.stream) {
subscribeWebsocketToChannel(session, firstParam(location.query.stream), location.query);
}
Expand Down

0 comments on commit aa10200

Please sign in to comment.