diff --git a/app/controllers/users_controller.rb b/app/controllers/users_controller.rb
index 78020fbb6c5..fdc76de103c 100644
--- a/app/controllers/users_controller.rb
+++ b/app/controllers/users_controller.rb
@@ -96,9 +96,16 @@ def impersonate
def invalidate_jwt
@user = find_resource(:edit_users)
@user.jwt_secret&.destroy
- process_success(
- :success_msg => _('Successfully invalidated JWTs for %s.') % @user.login
- )
+ respond_to do |format|
+ format.html do
+ process_success(
+ :success_msg => _('Successfully invalidated JWTs for %s.') % @user.login
+ )
+ end
+ format.json do
+ render :json => {}, :status => :ok
+ end
+ end
end
def stop_impersonation
diff --git a/app/views/jwt_tokens/_jwt_tokens_tab.html.erb b/app/views/jwt_tokens/_jwt_tokens_tab.html.erb
new file mode 100644
index 00000000000..212031b2fa4
--- /dev/null
+++ b/app/views/jwt_tokens/_jwt_tokens_tab.html.erb
@@ -0,0 +1,5 @@
+
<% caption = @user.inherited_admin? ? _('Admin rights are currently inherited from a user group') : '' %>
<%= checkbox_f f, :admin, help_block: caption if User.current.can_change_admin_flag? %>
diff --git a/test/controllers/users_controller_test.rb b/test/controllers/users_controller_test.rb
index 6b1a4bdae92..735f34bddd5 100644
--- a/test/controllers/users_controller_test.rb
+++ b/test/controllers/users_controller_test.rb
@@ -167,6 +167,15 @@ class UsersControllerTest < ActionController::TestCase
assert_nil user.jwt_secret
end
+ test "User should be able to invalidate jwt for self" do
+ User.current = users(:one)
+ user = User.current
+ FactoryBot.build(:jwt_secret, token: 'test_jwt_secret', user: user)
+ patch :invalidate_jwt, params: { :id => user.id }
+ user.reload
+ assert_nil user.jwt_secret
+ end
+
test 'user with edit users permission should be able to invalidate jwt for another user' do
User.current = setup_user "edit", "users"
user = users(:two)
diff --git a/webpack/assets/javascripts/react_app/components/componentRegistry.js b/webpack/assets/javascripts/react_app/components/componentRegistry.js
index 5e8c9a509a7..2e5b8e66c4d 100644
--- a/webpack/assets/javascripts/react_app/components/componentRegistry.js
+++ b/webpack/assets/javascripts/react_app/components/componentRegistry.js
@@ -44,6 +44,7 @@ import LabelIcon from './common/LabelIcon';
import { WelcomeAuthSource } from './AuthSource/Welcome';
import { WelcomeConfigReports } from './ConfigReports/Welcome';
import { WelcomeArchitecture } from './Architectures/Welcome';
+import JwtTokens from './users/JwtTokens/JwtTokens';
const componentRegistry = {
registry: forceSingleton('component_registry', () => ({})),
@@ -142,6 +143,7 @@ const coreComponents = [
{ name: 'SettingsTable', type: SettingsTable },
{ name: 'SettingUpdateModal', type: SettingUpdateModal },
{ name: 'PersonalAccessTokens', type: PersonalAccessTokens },
+ { name: 'JwtTokens', type: JwtTokens },
{ name: 'ClipboardCopy', type: ClipboardCopy },
{ name: 'LabelIcon', type: LabelIcon },
{
diff --git a/webpack/assets/javascripts/react_app/components/users/JwtTokens/JwtTokens.js b/webpack/assets/javascripts/react_app/components/users/JwtTokens/JwtTokens.js
new file mode 100644
index 00000000000..b1e126b31ab
--- /dev/null
+++ b/webpack/assets/javascripts/react_app/components/users/JwtTokens/JwtTokens.js
@@ -0,0 +1,64 @@
+import React, { Fragment } from 'react';
+import { Button } from '@patternfly/react-core';
+import { KeyIcon } from '@patternfly/react-icons';
+import { useDispatch } from 'react-redux';
+import PropTypes from 'prop-types';
+import { APIActions } from '../../../redux/API';
+import { openConfirmModal } from '../../ConfirmModal';
+import { translate as __ } from '../../../common/I18n';
+
+const JwtTokens = ({ userId }) => {
+ const dispatch = useDispatch();
+ return (
+
+
+
+
+
+
+
+
+ {__('JWT Tokens')}
+
+ {__(
+ 'By invalidating your JSON Web Tokens (JWTs), you will no longer be able to register hosts by using your existing JWTs.'
+ )}
+
+
+ |
+
+
+
+
+ );
+};
+
+JwtTokens.propTypes = {
+ userId: PropTypes.string.isRequired,
+};
+
+export default JwtTokens;