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

V4 API for run(), call(), spawn() #912

Open
cowboyd opened this issue Sep 26, 2024 · 8 comments
Open

V4 API for run(), call(), spawn() #912

cowboyd opened this issue Sep 26, 2024 · 8 comments
Milestone

Comments

@cowboyd
Copy link
Member

cowboyd commented Sep 26, 2024

One of the things happening in v4 is that we're aligning call() to be more similar to Function.prototype.call() This would imply that the signature for call() would be:

call(fn, thisObj, ...args)

The question is 1) is this desirable? and 2) what should we do with the other "call-like" functions in Effection: run() and spawn(). We could align them like this;

// v3
spawn(fn);
run(fn);

//v4?
spawn(fn, thisObj, ...args);
run(fn, thisObj, ...args);

On the one hand, this has a nice alignment with vanilla js, but on the other, it's annoying in that we don't really ever use this. Still, this is supposed to feel like native JS building blocks, so I'm not sure what the best way to go is.

On the other hand, I've been toying with the idea of deprecating scope.run() and replacing it by passing scope as a parameter to run() This would re-enforce the idea that this is an entry point from javascript and something you should do with care.

run(fn, { scope })

Which is better?

@iplaylf2
Copy link

it's annoying in that we don't really ever use this.

Conversely, when using NestJS and Effection together, I often need to use this, which can be quite inconvenient.

import { globalScope } from '..'

@Controller('receiver')
export class ReceiverController {
  @Inject()
  private readonly passport!: Passport
  
  @Inject()
  private readonly pushService!: PushService

  @Delete()
  public [`@Delete()`](): Promise<void> {
    return globalScope.run(function*(this: ReceiverController) {
      const receiver = yield * this.pushService.getClaimerReceiver(this.passport.id)
  
      if (null === receiver) {
        return
      }
  
      yield * this.pushService.deleteReceiver(receiver)
    }.bind(this))
  }
}

@cowboyd
Copy link
Member Author

cowboyd commented Oct 16, 2024

@iplaylf2 Thanks so much for your comment. This is a great data point!

@mikerourke
Copy link

I just came across Effection last week and am loving it! Just my two cents on this: you could use an approach similar to Redux-Saga and allow the first arg to call be function or an array where the first element of the array is the thisArg and the second is the function:

class SomeClass {
  public async doStuff(someArg: string): Promise<void> {
    // Stuff being done...
  }
}

const someThing = new SomeClass();

yield* call([someThing, someThing.doStuff], "Hello");

That would make it easier for users to upgrade to the new API and wouldn't require a lot of extra code on your part to check if the call function has a this context.

@cowboyd
Copy link
Member Author

cowboyd commented Nov 18, 2024

Glad to hear you like it @mikerourke! We've got a lot of fun stuff coming down the pike too.

This is a really interesting approach that I hadn't considered, but it seems really nice! Is there any downside in your experience? /cc @neurosnap

@mikerourke
Copy link

mikerourke commented Nov 18, 2024

@cowboyd I've been using Redux-Saga in several projects for the last 4 years, and I haven't encountered any issues with that API for call. It's probably worth mentioning that Redux-Saga also provides an apply effect that has a similar API to Function.prototype.apply that would look like this in Effection:

yield* apply(someThing, someThing.doStuff, ["Hello"]);

If you end up opting for the "array as a first arg for call" approach when you need a this context, it might be nice to provide apply as a convenience wrapper around the call([thisArg, fn], ...args) API.

@neurosnap
Copy link
Collaborator

The question is 1) is this desirable?

I think it depends. redux-saga call function type signature is pretty complicated since it can handle so many use cases -- including properly handling this. All of this is stemmed from the fact that the end-user doesn't call the function being passed into call, rather, the saga runtime handles calling all functions. The reason why this was preferred is because of the way saga handled side-effects: the end-user doesn't have to deal with managing side-effects because user functions don't actually call them.

The thing I liked about v3 call() is that we didn't have to deal with these use cases at all since the end-user does call the function itself: call(() => response.json());

So if all we are trying to do is align Function.prototype.call with v3 call() then I would rather argue we just change the name of call to something specific to what it is trying to accomplish in effection.

However, if in v4 we are moving to a world that is more similar to redux-saga where the effection runtime run() evaluates a bunch of yielded json objects and then converts them into function calls / activating side-effects, then we will most likely have to do it the way suggested above.

Lemme know if that doesn't make sense since this sort of relies on understanding how redux-saga works.

@mikerourke
Copy link

@neurosnap has an excellent point. When I first started reading the Effection docs, I drew a lot of parallels between Effection and redux-saga. But I'm just getting started with Effection, so I could be completely off base.

@cowboyd cowboyd added this to the v4 milestone Dec 9, 2024
@cowboyd
Copy link
Member Author

cowboyd commented Dec 13, 2024

Let's just keep it as function only, and we'll keep the rest of the API in our back pockets until we can make a clear determination. We can always add args to call((...args) => thing, ...args) if we start with call(() => thing)

That said, I do like using the array idea from redux saga as a way to pass this. I (think) we can add that in the future without making our call() overly complex.

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

No branches or pull requests

4 participants