## What problem are you trying to solve? At the moment, the way that `load` and `loadMany` are handled is very different. `load` is a regular promise that rejects, so awaits need to be `try-catch`ed. `loadMany` always resolves, so we need to iterate over the results in order to determine if we couldn't get anything. Since there's no indication in the Typescript declaration that one always resolves and the other doesn't. Not even a JSDoc to mention that if the batch function returns an error, the promise will reject. ## Describe the solution you'd like <!-- What should the API and behavior look like to address that problem? --> Unification in how we need to handle errors. Maybe something like Rust's `Result` type could be useful. [Neverthrow](https://github.com/supermacro/neverthrow) looks like a robust implementation of `Result` ## Describe alternatives you've considered <!-- Are there any alternative solutions or features you've considered that are inadequate for solving this problem? --> - Sticking with V1. Not as ideal, since the errors are useful for providing additional context to errors in a centralised location. Our custom error messages no longer need to include the entity ID. - Tried _several_ different ways of handling the errors, as detailed in the below additional context. We're struggling to come to a consensus on it, due to the fact that we need to be cognisant of the fact that `load` may throw if an ID isn't found in our DB. ## Additional context <!-- Any other context, examples, or adoptions paths to share? --> I'm having some trouble on my team trying to migrate to Dataloader v2. The improved error information is great, but migrating from V1 just feels unergonomic. Places where we would check for `null` are now utter chaos. We've tried try-catch blocks but the sheer amount of boilerplate that it introduces (can't use `const x = loader.load(...)` anymore) is unergonomic and verbose. We've tried promises, which feels ok at the loader callsite, since Prettier wraps the line nicely. But the wrappers are a bit unsightly with `wrapWithContextError` being a curryied function. ```js await loader .loadMany(idArray) .then(throwIfError) .catch(wrapWithContextError(wrapperError)) ``` The fact that `loader.load` needs to be handled significantly differently to `loader.loadMany` has brought with it some confusion. Here's the `load` variants ```js // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= // =-=-=- Allow the error to propagate =-=-=-= // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= const entity = await ctx.loaders.entityLoader.load(id); // Do something with the entity // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= // =-=-=- Allow the error to propagate, but wrap in a custom error =-=-=-= // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= let entity; try { entity = await ctx.loaders.entityLoader.load(id); } catch (e) { throw new UserInputError(`Could not find <entity name>`, { cause: e }); } // Do something with the entity // Or const entity = await ctx.loaders.entityLoader.load(id).catch(e => { throw new UserInputError(`Could not find <entity name>`, { cause: e }); }); // Do something with the entity // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= // =-=-=- Promises, ignoring the error =-=-=-= // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= const entity = await ctx.loaders.entityLoader.load(id).catch(() => null); // Do something with the entity ``` Here's the `loadMany` variants ```js // =-=-=-=-=-=-=-=-=-=-=-= // =-=-=- Promises =-=-=-= // =-=-=-=-=-=-=-=-=-=-=-= const entities = await ctx.loaders.entityLoader .loadMany(ids) .then(throwIfError()); // Do something with the collection // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= // =-=-=- Promises with a custom error =-=-=-= // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= const entities = await ctx.loaders.entityLoader .loadMany(ids) .then(throwIfError()) .catch(wrapWithContextError( new UserInputError("One or more <entity name>s could not be found") )); // Do something with the collection // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= // =-=-=- Promises, ignoring the error =-=-=-= // =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= const entities = await ctx.loaders.entityLoader .loadMany(ids) .then(entities => entities.map( entity => isError(entity) ? null : entity) ); // Do something with the collection // =-=-=-=-=-=-=-=-=-=-=-=-= // =-=-=- Type guard =-=-=-= // =-=-=-=-=-=-=-=-=-=-=-=-= const entities = await ctx.loaders.entityLoader.loadMany(ids); if (!loadSuccessful(entities)) { throw entities.find(isError); } // Do something with the collection ```