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

API improvements #482

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

API improvements #482

wants to merge 5 commits into from

Conversation

Legoless
Copy link

@Legoless Legoless commented Jul 30, 2024

  • A description about what and why you are contributing, even if it's trivial.

This PR adds two API improvements:

  1. Purchases Listener is currently an abstract MonoBehaviour, which ties the behaviour to Unity components. This is not bad, but it does limit you, which is entirely not needed, if you mostly use the API from code. There's no other requirement that this should in fact be an abstract class (aside from the ability to set it in Editor). So a new interface is introduced instead and backwards support is added, so that the UpdatedCustomerInfoListener implements the interface. Field is now set as private and serialized, so it can remain set in Editor, but via code Listener property should be used.

  2. The whole Purchases object which is required for RevenueCat to operate initializes itself fairly late in Start function. This is a problem, if you need to do anything beforehand as we're booting our game up in Awake method. The script order is usually undefined and we've seen many differences in it between real devices and Editor, so sometimes Purchases would prepare itself for API type of configuration, sometimes we would be too early calling Configure and crash with NullPointerException on internal _wrapper. To get around this problem, we extracted initialization of wrapper to a separate function that only creates the instances once, but can be called many times. That way Configure will Prepare the wrapper, before it calls to it's Setup method.

  • The issue number(s) or PR number(s) in the description if you are contributing in response to those.
    No open issues right now for these improvements.

  • If applicable, unit tests.
    No additional Unit Tests required.

Copy link
Contributor

@tonidero tonidero left a comment

Choose a reason for hiding this comment

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

Hi @Legoless,

Really sorry for the delay, and thanks for your contribution! The changes here make a lot of sense. I just left a comment about a possible breaking change. What do you think?

Once we get that sorted I can get this merged


// Cache custom listener, if not from class.
private IUpdatedCustomerInfoListener customListener;
public IUpdatedCustomerInfoListener Listener
Copy link
Contributor

Choose a reason for hiding this comment

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

I believe this means that devs setting the listener through code before might experience a breaking change.

  • Before:
purchases.listener = CustomListener()
  • After:
purchases.Listener = CustomListener()

I do think that most people would set the listener through the editor, considering it had to be a MonoBehavior... But I think we might need to make the listener public private still to avoid the breaking change, and encourage people to move to use Listener instead. Wdyt?

Copy link
Contributor

Choose a reason for hiding this comment

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

Maybe we can hold on this change until the next major as well, and release the change below

Copy link
Author

@Legoless Legoless Aug 13, 2024

Choose a reason for hiding this comment

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

You're correct, this is breaking change in code. I decided for this approach for two reasons:

  • It's very easy to search and replace in code: Listener instead of listener. Could leave some comments and hint maybe for those people that use it in code. It's unlikely people would be setting and resetting it a lot of times either.
  • It's very clear from API design perspective that the private field is meant for Editor only and you only allow setting it there. In other way you would have two exposed listener and Listener in the API, which could get confusing, if in fact you did want to set it in code as a new user.

My suggestion would be to keep the API clearer and rather make a major release (or however you think is best). If this breaking change is a problem, then this approach will make the API clearer in the future. But at the end it's your call how you want to have it, definitely both solutions work for our case. Maybe a better approach would be to have a List of those listeners instead and keep the original one public and just add like a AddListener/RemoveListener methods to manage the list.. Just thinking out loud.

Your call, let me know if you need me to change in the PR.

Copy link
Contributor

Choose a reason for hiding this comment

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

Yeah I agree that it would be better to ship this in a major, rather than keeping both which would cause a very confusing API.

If you want to revert this change and leave the one below, we would be happy to get that in!

As for your proposal of having AddListener/RemoveListener`, I think that could work! We actually allow doing this in some other SDKs. But seems that could be worked on separately. Thanks for the suggestion!

Copy link
Author

Choose a reason for hiding this comment

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

I can also do the Add/RemoveListener if you like, but not sure which change you want me to revert? The change to private?

Copy link
Contributor

Choose a reason for hiding this comment

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

Sorry for not being clear! My suggestion is to:

  • Revert the changes related to the listener (including the new interface)
  • Keep the changes related to initialization of the SDK at different points of time.

Then merge the PR with just that second change, which shouldn't be a breaking change. And we can release that soon.

As for doing the Add/Remove listener, I think that works too! And it would be amazing if you can do it 👼. The only point is to avoid breaking changes (at least until we're ready to make another major)

Please let me know if you have questions or help with the listener approach! We also think that would be a good addition 👍

Copy link
Author

Choose a reason for hiding this comment

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

Cool, I understand, I'll get on it this week.


// Cache custom listener, if not from class.
private IUpdatedCustomerInfoListener customListener;
public IUpdatedCustomerInfoListener Listener
Copy link
Contributor

Choose a reason for hiding this comment

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

Maybe we can hold on this change until the next major as well, and release the change below

@@ -111,22 +130,32 @@ public partial class Purchases : MonoBehaviour

private void Start()
{
#if UNITY_ANDROID && !UNITY_EDITOR
Prepare();
Copy link
Contributor

Choose a reason for hiding this comment

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

This is great! Makes a lot of sense, thanks for adding this!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants