Skip to content

Commit

Permalink
Fixes #37936 - Invalidate jwt for any user or users(API)
Browse files Browse the repository at this point in the history
  • Loading branch information
girijaasoni committed Dec 24, 2024
1 parent bc2dbbf commit ee4ccc7
Show file tree
Hide file tree
Showing 5 changed files with 108 additions and 1 deletion.
58 changes: 58 additions & 0 deletions app/controllers/api/v2/registration_tokens_controller.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
module Api
module V2
class RegistrationTokensController < V2::BaseController
include Foreman::Controller::UsersMixin

include Foreman::Controller::Parameters::User
before_action :authenticate, :only => [:invalidate_tokens]

def resource_class
User
end

def find_resource(permission = :view_users)
editing_self? ? User.find(User.current.id) : User.authorized(permission).except_hidden.find(params[:id])
end

def action_permission
case params[:action]
when 'invalidate_tokens', 'invalidate_jwt'
'edit'
else
super
end
end

api :DELETE, '/users/:id/registration_tokens', N_("Invalidate all registration Tokens for a specific user.")
description <<-DOC
The user you specify will no longer be able to register hosts by using their JWTs.
DOC
param :id, String, :desc => N_("ID of the user"), :required => true

def invalidate_jwt
@user = find_resource(:edit_users)
unless @user
raise ::Foreman::Exception.new(N_("No record found for %s"), params[:id])
end
@user.jwt_secret&.destroy
process_success _('Successfully invalidated JWTs for %s.' % @user.login)
end

api :DELETE, "/registration_tokens", N_("Invalidate all JSON Web Tokens (JWTs) for multiple users.")
param :search, String, :required => true
description <<-DOC
The users you specify will no longer be able to register hosts by using their JWTs.
DOC

def invalidate_tokens
raise ::Foreman::Exception.new(N_("Please provide search parameters")) if params[:search].blank?
@users = resource_scope_for_index(:permission => :edit_users).except_hidden.uniq
if @users.blank?
raise ::Foreman::Exception.new(N_("No record found %s"), params[:search])
end
JwtSecret.where(user_id: @users).destroy_all
process_success _('Successfully invalidated JWTs for %s.' % @users.pluck(:login).to_sentence)
end
end
end
end
3 changes: 3 additions & 0 deletions app/models/user.rb
Original file line number Diff line number Diff line change
Expand Up @@ -476,6 +476,9 @@ def editing_self?(options = {})
options[:user_id].to_i == id ||
options[:controller].to_s == 'api/v2/personal_access_tokens' &&
options[:action] =~ /show|destroy|index|create/ &&
options[:user_id].to_i == id ||
options[:controller].to_s == 'api/v2/registration_tokens' &&
options[:action] =~ /invalidate_jwt|invalidate_tokens/ &&
options[:user_id].to_i == id
end

Expand Down
3 changes: 2 additions & 1 deletion config/initializers/f_foreman_permissions.rb
Original file line number Diff line number Diff line change
Expand Up @@ -563,7 +563,8 @@
:"api/v2/users" => [:create]
map.permission :edit_users,
:users => [:edit, :update, :invalidate_jwt],
:"api/v2/users" => [:update]
:"api/v2/users" => [:update],
:"api/v2/registration_tokens" => [:invalidate_tokens, :invalidate_jwt]
map.permission :destroy_users,
:users => [:destroy],
:"api/v2/users" => [:destroy]
Expand Down
7 changes: 7 additions & 0 deletions config/routes/api/v2.rb
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,13 @@
resources :mail_notifications, :only => [:create, :destroy, :update]
get 'mail_notifications', :to => 'mail_notifications#user_mail_notifications', :on => :member
get 'extlogin', :to => 'users#extlogin', :on => :collection
delete 'registration_tokens', :to => 'registration_tokens#invalidate_jwt', :on => :member
end
end

resources :registration_tokens, :only => [:invalidate_tokens] do
collection do
delete '/', :action => :invalidate_tokens
end
end

Expand Down
38 changes: 38 additions & 0 deletions test/controllers/api/v2/registration_tokens_controller_test.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
require 'test_helper'

class Api::V2::RegistrationTokensControllerTest < ActionController::TestCase
test 'user shall invalidate tokens for self' do
user = users(:one)
FactoryBot.create(:jwt_secret, token: 'test_jwt_secret', user: user)
delete :invalidate_jwt, params: { :id => user.id.to_s}, session: set_session_user(user)
user.reload
assert_nil user.jwt_secret
end

test 'user with edit permission should be able to invalidate jwt for another user' do
setup_user 'edit', 'users'
user = users(:one)
FactoryBot.create(:jwt_secret, token: 'test_jwt_secret', user: user)
delete :invalidate_tokens, params: { :search => "id ^ (#{user.id})"}
user.reload
assert_nil user.jwt_secret
end

test 'user without edit permission should not be able to invalidate jwt for another user' do
User.current = users(:one)
user = users(:two)
FactoryBot.create(:jwt_secret, token: 'test_jwt_secret', user: user)
delete :invalidate_tokens, params: { :search => "id ^ (#{user.id})"}, session: set_session_user(User.current)
user.reload
assert_response :error

Check failure on line 27 in test/controllers/api/v2/registration_tokens_controller_test.rb

View workflow job for this annotation

GitHub Actions / test:functionals - Ruby 3.0 and Node 14 on PostgreSQL 13

Failure: test_0003_user without edit permission should not be able to invalidate jwt for another user Expected response to be a <5XX: error>, but was a <403: Forbidden> Response body: { "error": {"message":"Access denied","details":"Missing one of the required permissions: edit_users","missing_permissions":["edit_users"]} }

Check failure on line 27 in test/controllers/api/v2/registration_tokens_controller_test.rb

View workflow job for this annotation

GitHub Actions / test:functionals - Ruby 2.7 and Node 14 on PostgreSQL 13

Failure: test_0003_user without edit permission should not be able to invalidate jwt for another user Expected response to be a <5XX: error>, but was a <403: Forbidden> Response body: { "error": {"message":"Access denied","details":"Missing one of the required permissions: edit_users","missing_permissions":["edit_users"]} }
end

test 'invalidating jwt should fail without search params' do
setup_user 'edit', 'users'
user = users(:two)
FactoryBot.create(:jwt_secret, token: 'test_jwt_secret', user: user)
delete :invalidate_tokens, session: set_session_user(User.current)
user.reload
assert_response :error
end
end

0 comments on commit ee4ccc7

Please sign in to comment.