fastify-basic-auth
fastify-basic-auth copied to clipboard
Fix validate() return type
Checklist
- [x] run
npm run test
andnpm run benchmark
- [x] tests and/or benchmarks are included
- [ ] documentation is changed or added
- [x] commit message and code follows the Developer's Certification of Origin and the Code of conduct
all done. I also completely split the possible function signatures.
Why did the tests not run?
They did.
There are errors:
index.test-d.ts:16:4
✖ 15:44 Parameter username implicitly has an any type.
✖ 15:54 Parameter password implicitly has an any type.
✖ 15:64 Parameter req implicitly has an any type.
✖ 15:69 Parameter reply implicitly has an any type.
✖ 16:4 Parameter type string is not identical to argument type any.
✖ 17:4 Parameter type string is not identical to argument type any.
✖ 18:4 Parameter type FastifyRequest<RouteGenericInterface, Server, IncomingMessage> is not identical to argument type any.
✖ 19:4 Parameter type FastifyReply<Server, IncomingMessage, ServerResponse, RouteGenericInterface, unknown> is not identical to argument type any.
8 errors
Ah, I guess it’s this one: https://github.com/microsoft/TypeScript/issues/598
This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions.
So how do we fix this? I can’t write a test if TypeScript bugs out on me
cc @fastify/typescript can you help?
@flying-sheep you need to address the type checker error, e.g change validatePromise (username, password, req, reply)
to validatePromise (username: string, password: string, req: FastifyRequest, reply: FastifyReply)
Hmm, I thought the whole point of the test was to check if the types get inferred properly …
Wow, sorry, you're right. Looks like an odd TS behaviour.
Not sure what's best here. Perhaps export two definitions?
export type FastifyBasicAuthValidateFnCallback = (username: string, password: string, req: FastifyRequest, reply: FastifyReply, done: (err?: Error) => void) => void
export type FastifyBasicAuthValidateFnPromise = (username: string, password: string, req: FastifyRequest, reply: FastifyReply) => Promise<Error|void>
export interface FastifyBasicAuthOptions {
validate: FastifyBasicAuthValidateFnPromise|FastifyBasicAuthValidateFnCallback;
authenticate?: boolean | { realm: string };
header?: string;
}
This way you can do something like this?
app.register(fastifyBasicAuth, {
validate: function (username, password, req, reply) {
return new Promise((resolve, reject) => resolve())
} as FastifyBasicAuthValidateFnPromise
})
Maybe there's a better solution? I quickly tried conditional types (to return void or a Promise) based on the presence of done
, but had some issues getting that to work.
Welp, maybe not possible then. Too bad.
Yeah, seems like it's necessary to explicitly define the type for the Promise-based approach. Exporting those FastifyBasicAuthValidateFnPromise
and FastifyBasicAuthValidateFnCallback
seems like an alright compromise perhaps?
At least it will provide a hint via intellisense like so:
data:image/s3,"s3://crabby-images/57e0e/57e0eb7f1ff89ddde05a588d008b7a7b18e42b36" alt="Screenshot 2022-04-21 at 1 19 46 PM"
The test case could become:
app.register(fastifyBasicAuth, {
validate: function validateCallback (username, password, req, reply) {
expectType<string>(username)
expectType<string>(password)
expectType<FastifyRequest>(req)
expectType<FastifyReply>(reply)
return new Promise<void>((resolve, reject) => {resolve()})
} as FastifyBasicAuthValidateFnPromise,
header: 'x-forwarded-authorization'
})
Maybe we can get a TypeScript wiz in here to help. It would be great to get it to a state where it’s correct but TypeScript can also infer FastifyBasicAuthValidateFnPromise
. As it stands, users have to manually say that, which would be a regression.
I tried this, but it didn’t change anything:
type IsArg5Valid<T extends (...a: any) => any, E> = Parameters<T>['length'] extends 5 ? {} : E;
export interface FastifyBasicAuthOptions {
validate:
FastifyBasicAuthValidateFnPromise
| (FastifyBasicAuthValidateFnCallback & IsArg5Valid<FastifyBasicAuthValidateFnCallback, "You need to specify a “done” callback">);
...
}
I looked into this and I am not sure what is going on. The main issue might be on the FastifyPlugin
generic side. I'll look deeper into this ASAP. Thanks for the PR.
This isn't specific to FastifyPlugin
AFAICT. Just a quirk of TS. Take a look at this playground example that exhibits the same behaviour. A TypeScript wizard can possibly figure out some clever trick though!