我已经研究了这两者passport-facebook
以及passport-facebook-token
与 NestJS 的集成。问题在于 NestJS 使用自己的实用程序(例如 AuthGuard)抽象了护照实现。
因此,ExpressJS
文档中的样式实现不适用于 NestJS。@nestjs/passport
例如,这与软件包不兼容:
var FacebookTokenStrategy = require('passport-facebook-token');
passport.use(new FacebookTokenStrategy({
clientID: FACEBOOK_APP_ID,
clientSecret: FACEBOOK_APP_SECRET
}, function(accessToken, refreshToken, profile, done) {
User.findOrCreate({facebookId: profile.id}, function (error, user) {
return done(error, user);
});
}
));
这篇博文展示了一种passport-facebook-token
使用不熟悉的、不符合AuthGuard
.
@Injectable()
export class FacebookStrategy {
constructor(
private readonly userService: UserService,
) {
this.init();
}
init() {
use(
new FacebookTokenStrategy(
{
clientID: <YOUR_APP_CLIENT_ID>,
clientSecret: <YOUR_APP_CLIENT_SECRET>,
fbGraphVersion: 'v3.0',
},
async (
accessToken: string,
refreshToken: string,
profile: any,
done: any,
) => {
const user = await this.userService.findOrCreate(
profile,
);
return done(null, user);
},
),
);
}
}
这里的问题是,这似乎与 NestJS 期望您处理护照策略的方式完全不同。它是一起被黑的。它也可能在未来的 NestJS 更新中中断。这里也没有异常处理;我无法捕获异常,例如由于正在使用的回调性质而InternalOAuthError
引发的异常。passport-facebook-token
是否有一种干净的方法来实现其中一个passport-facebook
或passport-facebook-token
以便它使用@nestjs/passport
'validate()
方法?来自文档:对于每个策略,Passport 将调用验证函数(使用 @nestjs/passport 中的 validate() 方法实现)。应该有一种方法可以在构造函数中传递 a clientId
,clientSecret
然后将其余逻辑放入validate()
方法中。
我想最终结果看起来类似于以下内容(这不起作用):
import { Injectable } from "@nestjs/common";
import { PassportStrategy } from "@nestjs/passport";
import FacebookTokenStrategy from "passport-facebook-token";
@Injectable()
export class FacebookStrategy extends PassportStrategy(FacebookTokenStrategy, 'facebook')
{
constructor()
{
super({
clientID : 'anid', // <- Replace this with your client id
clientSecret: 'secret', // <- Replace this with your client secret
})
}
async validate(request: any, accessToken: string, refreshToken: string, profile: any, done: Function)
{
try
{
console.log(`hey we got a profile: `, profile);
const jwt: string = 'placeholderJWT'
const user =
{
jwt
}
done(null, user);
}
catch(err)
{
console.log(`got an error: `, err)
done(err, false);
}
}
}
在我的特殊情况下,我对callbackURL
. 我只是在验证客户端已转发到服务器的访问令牌。我只是把上面说得很清楚。
另外,如果您好奇,上面的代码会产生一个,InternalOAuthError
但我无法在策略中捕获异常以查看真正的问题是什么,因为它没有正确实现。我知道在这种特殊情况下,access_token
我传递的是无效的,如果我传递一个有效的,代码就可以工作。通过正确的实现,虽然我将能够捕获异常,检查错误,并能够向用户发出适当的异常,在本例中为 HTTP 401。
InternalOAuthError: Failed to fetch user profile
很明显,异常是在validate()
方法之外抛出的,这就是为什么我们的 try/catch 块没有捕获InternalOAuthError
. 处理此异常对于正常的用户体验至关重要,我不确定在此实现中 NestJS 处理它的方式是什么,或者应该如何进行错误处理。