Skip to content

Commit

Permalink
HADOOP-17159. Make UGI support forceful relogin from keytab ignoring …
Browse files Browse the repository at this point in the history
…the last login time (apache#2249)

Contributed by Sandeep Guggilam.

Signed-off-by: Mingliang Liu <[email protected]>
Signed-off-by: Steve Loughran <[email protected]>
  • Loading branch information
sguggilam authored and PACordonnier committed Nov 7, 2023
1 parent 3e9e04d commit ecf1fd0
Show file tree
Hide file tree
Showing 2 changed files with 66 additions and 6 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -1107,7 +1107,29 @@ public void reloginFromKeytab() throws IOException {
reloginFromKeytab(false);
}

/**
* Force re-Login a user in from a keytab file irrespective of the last login
* time. Loads a user identity from a keytab file and logs them in. They
* become the currently logged-in user. This method assumes that
* {@link #loginUserFromKeytab(String, String)} had happened already. The
* Subject field of this UserGroupInformation object is updated to have the
* new credentials.
*
* @throws IOException
* @throws KerberosAuthException on a failure
*/
@InterfaceAudience.Public
@InterfaceStability.Evolving
public void forceReloginFromKeytab() throws IOException {
reloginFromKeytab(false, true);
}

private void reloginFromKeytab(boolean checkTGT) throws IOException {
reloginFromKeytab(checkTGT, false);
}

private void reloginFromKeytab(boolean checkTGT, boolean ignoreLastLoginTime)
throws IOException {
if (!shouldRelogin() || !isFromKeytab()) {
return;
}
Expand All @@ -1122,7 +1144,7 @@ private void reloginFromKeytab(boolean checkTGT) throws IOException {
return;
}
}
relogin(login);
relogin(login, ignoreLastLoginTime);
}

/**
Expand All @@ -1143,25 +1165,27 @@ public void reloginFromTicketCache() throws IOException {
if (login == null) {
throw new KerberosAuthException(MUST_FIRST_LOGIN);
}
relogin(login);
relogin(login, false);
}

private void relogin(HadoopLoginContext login) throws IOException {
private void relogin(HadoopLoginContext login, boolean ignoreLastLoginTime)
throws IOException {
// ensure the relogin is atomic to avoid leaving credentials in an
// inconsistent state. prevents other ugi instances, SASL, and SPNEGO
// from accessing or altering credentials during the relogin.
synchronized(login.getSubjectLock()) {
// another racing thread may have beat us to the relogin.
if (login == getLogin()) {
unprotectedRelogin(login);
unprotectedRelogin(login, ignoreLastLoginTime);
}
}
}

private void unprotectedRelogin(HadoopLoginContext login) throws IOException {
private void unprotectedRelogin(HadoopLoginContext login,
boolean ignoreLastLoginTime) throws IOException {
assert Thread.holdsLock(login.getSubjectLock());
long now = Time.now();
if (!hasSufficientTimeElapsed(now)) {
if (!hasSufficientTimeElapsed(now) && !ignoreLastLoginTime) {
return;
}
// register most recent relogin attempt
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -127,6 +127,42 @@ public void testUGILoginFromKeytab() throws Exception {
Assert.assertNotSame(login1, login2);
}

/**
* Force re-login from keytab using the MiniKDC and verify the UGI can
* successfully relogin from keytab as well.
*/
@Test
public void testUGIForceReLoginFromKeytab() throws Exception {
// Set this to false as we are testing force re-login anyways
UserGroupInformation.setShouldRenewImmediatelyForTests(false);
String principal = "foo";
File keytab = new File(workDir, "foo.keytab");
kdc.createPrincipal(keytab, principal);

UserGroupInformation.loginUserFromKeytab(principal, keytab.getPath());
UserGroupInformation ugi = UserGroupInformation.getLoginUser();
Assert.assertTrue("UGI should be configured to login from keytab",
ugi.isFromKeytab());

// Verify relogin from keytab.
User user = getUser(ugi.getSubject());
final long firstLogin = user.getLastLogin();
final LoginContext login1 = user.getLogin();
Assert.assertNotNull(login1);

// Sleep for 2 secs to have a difference between first and second login
Thread.sleep(2000);

// Force relogin from keytab
ugi.forceReloginFromKeytab();
final long secondLogin = user.getLastLogin();
final LoginContext login2 = user.getLogin();
Assert.assertTrue("User should have been able to relogin from keytab",
secondLogin > firstLogin);
Assert.assertNotNull(login2);
Assert.assertNotSame(login1, login2);
}

@Test
public void testGetUGIFromKnownSubject() throws Exception {
KerberosPrincipal principal = new KerberosPrincipal("user");
Expand Down

0 comments on commit ecf1fd0

Please sign in to comment.