Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Core: Simplify AuthManager API #12555

Open
wants to merge 9 commits into
base: main
Choose a base branch
from

Conversation

adutra
Copy link
Contributor

@adutra adutra commented Mar 17, 2025

The current AuthManager interface exposes various methods for creating sessions: initSession(), catalogSession(), contextualSession(), tableSession(), etc.

Naming its methods after the intended usage of the returned session was, in hindsight, a questionable design choice: in some cases, the right method to call may not be so obvious, and if more session "flavors" are need in the future, more methods would need to be added to the interface.

This PR aims at fixing this by unifying all previous methods under one single method. The increased flexibility of the new method is due to its argument of type AuthScope, which can be freely implemented according to the caller's needs.

@adutra adutra force-pushed the auth-manager-api-redesign branch 2 times, most recently from 9ff0603 to 5a9aa92 Compare March 18, 2025 00:12
@ajantha-bhat ajantha-bhat modified the milestone: Iceberg 1.9.0 Mar 18, 2025
@Override
@Deprecated
@SuppressWarnings("deprecation")
public OAuth2Util.AuthSession catalogSession(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

shouldn't all of these overrides rather go into AuthManager where the default impl would be to call what's being called here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can't remove these methods without introducing revapi exceptions.

If you are OK with revapi exceptions, then I'm fine with that too!

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what I mean is this: the AuthManager already deprecated those methods and here we're just overriding them for the sake of providing a slightly different impl. IMO that impl should go into the AuthManager so that we don't need to do anything for those deprecated methods here

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

If I remove the methods here I get many revapi error like this:

java.method.returnTypeChanged: The return type changed from 'org.apache.iceberg.rest.auth.OAuth2Util.AuthSession' to  'org.apache.iceberg.rest.auth.AuthSession'.
  
  old: method org.apache.iceberg.rest.auth.OAuth2Util.AuthSession org.apache.iceberg.rest.auth.OAuth2Manager::catalogSession(org.apache.iceberg.rest.RESTClient, java.util.Map<java.lang.String, java.lang.String>)
  new: method org.apache.iceberg.rest.auth.AuthSession org.apache.iceberg.rest.auth.AuthManager::catalogSession(org.apache.iceberg.rest.RESTClient, java.util.Map<java.lang.String, java.lang.String>) @ org.apache.iceberg.rest.auth.OAuth2Manager

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tried something in 75acdbc – let me know if that's better.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

hm ok I was hoping that RevAPI wouldn't complain. The changes introduced in that commit make sense to me, thanks

@@ -30,7 +30,7 @@ class TestBasicAuthManager {
@Test
void missingUsername() {
try (AuthManager authManager = new BasicAuthManager("test")) {
assertThatThrownBy(() -> authManager.catalogSession(null, Map.of()))
assertThatThrownBy(() -> authManager.authSession(ImmutableAuthScopes.Standalone.of(Map.of())))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

so from how I understand it, the Standalone scope is basically the same as the Catalog scope and only differs in naming?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Mostly, yes. I tried to find a name that would capture its "generic" purpose better than just calling it Catalog or Table.

@Override
@Deprecated
@SuppressWarnings("deprecation")
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think this is necessary. All the other components that we deprecate don't suppress this warning, since we want to make sure that users see this deprecation and switch to the new API

@Value.Immutable
abstract class Initial implements AuthScope {

@Value.Parameter(order = 1)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not a big fan of using @Value.Parameter to have an of() method being generated with the correct paramters. In all other places we typically create the of() method ourselves to better control the signature of the method. This also makes it easier for us down the line in case anything changes with the parameters, since we can properly deprecate and create overloading methods when needed. When the of() method is generated, it's more difficult to go through our API deprecation cycle.

That being said, I would change this to

diff --git a/core/src/main/java/org/apache/iceberg/rest/auth/AuthScopes.java b/core/src/main/java/org/apache/iceberg/rest/auth/AuthScopes.java
index 4465c95290..c9ab1fc5e3 100644
--- a/core/src/main/java/org/apache/iceberg/rest/auth/AuthScopes.java
+++ b/core/src/main/java/org/apache/iceberg/rest/auth/AuthScopes.java
@@ -36,7 +36,6 @@ public interface AuthScopes {
   @Value.Immutable
   abstract class Initial implements AuthScope {
 
-    @Value.Parameter(order = 1)
     @Override
     public abstract Map<String, String> properties();
 
@@ -49,6 +48,10 @@ public interface AuthScopes {
     public final boolean cacheable() {
       return false;
     }
+
+    public static Initial of(Map<String, String> properties) {
+      return ImmutableAuthScopes.Initial.builder().properties(properties).build();
+    }
   }
 
   /**
@@ -58,7 +61,6 @@ public interface AuthScopes {
   @Value.Immutable
   abstract class Catalog implements AuthScope {
 
-    @Value.Parameter(order = 1)
     @Override
     public abstract Map<String, String> properties();
 
@@ -71,6 +73,10 @@ public interface AuthScopes {
     public final boolean cacheable() {
       return false;
     }
+
+    public static Catalog of(Map<String, String> properties) {
+      return ImmutableAuthScopes.Catalog.builder().properties(properties).build();
+    }
   }
 
   /**
@@ -80,10 +86,8 @@ public interface AuthScopes {
   @Value.Immutable
   abstract class Contextual implements AuthScope {
 
-    @Value.Parameter(order = 1)
     public abstract SessionCatalog.SessionContext context();
 
-    @Value.Parameter(order = 2)
     @Value.Lazy
     @Override
     public Map<String, String> properties() {
@@ -94,7 +98,6 @@ public interface AuthScopes {
       return RESTUtil.merge(properties, credentials);
     }
 
-    @Value.Parameter(order = 3)
     @Override
     @Nonnull
     public abstract AuthSession parent();
@@ -103,6 +106,10 @@ public interface AuthScopes {
     public final boolean cacheable() {
       return true;
     }
+
+    public static Contextual of(SessionCatalog.SessionContext context, AuthSession parent) {
+      return ImmutableAuthScopes.Contextual.builder().context(context).parent(parent).build();
+    }
   }
 
   /**
@@ -112,14 +119,11 @@ public interface AuthScopes {
   @Value.Immutable
   abstract class Table implements AuthScope {
 
-    @Value.Parameter(order = 1)
     public abstract TableIdentifier identifier();
 
-    @Value.Parameter(order = 2)
     @Override
     public abstract Map<String, String> properties();
 
-    @Value.Parameter(order = 3)
     @Override
     @Nonnull
     public abstract AuthSession parent();
@@ -128,6 +132,15 @@ public interface AuthScopes {
     public final boolean cacheable() {
       return true;
     }
+
+    public static Table of(
+        TableIdentifier identifier, Map<String, String> properties, AuthSession parent) {
+      return ImmutableAuthScopes.Table.builder()
+          .identifier(identifier)
+          .properties(properties)
+          .parent(parent)
+          .build();
+    }
   }
 
   /**
@@ -137,7 +150,6 @@ public interface AuthScopes {
   @Value.Immutable
   abstract class Standalone implements AuthScope {
 
-    @Value.Parameter(order = 1)
     @Override
     public abstract Map<String, String> properties();
 
@@ -146,5 +158,9 @@ public interface AuthScopes {
     public boolean cacheable() {
       return false;
     }
+
+    public static Standalone of(Map<String, String> properties) {
+      return ImmutableAuthScopes.Standalone.builder().properties(properties).build();
+    }
   }
 }

And when a scope is instantiated I would use AuthScopes.Xyz.of(..) instead of ImmutableAuthScopes.Xyz.of(..)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Very nice suggestion, patch applied 👍

this.name = managerName;
}

@Override
public OAuth2Manager withClient(RESTClient restClient) {
Preconditions.checkNotNull(restClient, "invalid restClient: null");
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this should probably be a IAE to indicate that the argument passed cannot be null. Also I would update the error msg to Invalid REST client: null


@Override
public OAuth2Util.AuthSession authSession(AuthScope scope) {
Preconditions.checkNotNull(scope, "invalid scope: null");
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
Preconditions.checkNotNull(scope, "invalid scope: null");
Preconditions.checkArgument(null != scope, "Invalid auth scope: null");

@@ -130,24 +136,26 @@ void catalogSessionNoOAuth2Properties() {
@Test
void catalogSessionTokenProvided() {
Map<String, String> properties = Map.of(OAuth2Properties.TOKEN, "test");
try (OAuth2Manager manager = new OAuth2Manager("test");
OAuth2Util.AuthSession catalogSession = manager.catalogSession(client, properties)) {
try (OAuth2Manager manager = new OAuth2Manager("test").withClient(client).withClient(client);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
try (OAuth2Manager manager = new OAuth2Manager("test").withClient(client).withClient(client);
try (OAuth2Manager manager = new OAuth2Manager("test").withClient(client);

adutra added 6 commits March 26, 2025 10:14
The current `AuthManager` interface exposes various methods for creating sessions: `initSession()`, `catalogSession()`, `contextualSession()`, `tableSession()`, etc.

Naming its methods after the intended usage of the returned session was, in hindsight, a questionable design choice: in some cases, the right method to call may not be so obvious, and if more session "flavors" are need in the future, more methods would need to be added to the interface.

This PR aims at fixing this by unifying all previous methods under one single method. The increased flexibility of the new method is due to its argument of type `AuthScope`, which can be freely implemented according to the caller's needs.
@adutra adutra force-pushed the auth-manager-api-redesign branch from 041bcd0 to 414291e Compare March 26, 2025 09:24
}

/**
* Returns a long-lived session whose lifetime is tied to the owning catalog. This session serves
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

minor: I don't think it's necessary to remove all javadoc just because we're deprecating something. I would just leave it in, since that also reduces the size of the diff

@adutra
Copy link
Contributor Author

adutra commented Mar 28, 2025

@nastra is there any pending issues in this PR?

@nastra nastra requested a review from danielcweeks March 28, 2025 18:15
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants