2022-04-28 02:55:49 -05:00
|
|
|
import { ConnectorError, ConnectorErrorCodes, GetConnectorConfig } from '@logto/connector-types';
|
|
|
|
import nock from 'nock';
|
|
|
|
|
|
|
|
import { WeChatNativeConnector } from '.';
|
2022-05-13 01:22:37 -05:00
|
|
|
import { accessTokenEndpoint, authorizationEndpoint, userInfoEndpoint } from './constant';
|
2022-04-28 02:55:49 -05:00
|
|
|
import { mockedConfig } from './mock';
|
2022-05-13 01:22:37 -05:00
|
|
|
import { WeChatNativeConfig } from './types';
|
2022-04-28 02:55:49 -05:00
|
|
|
|
|
|
|
const getConnectorConfig = jest.fn() as GetConnectorConfig<WeChatNativeConfig>;
|
|
|
|
|
|
|
|
const weChatNativeMethods = new WeChatNativeConnector(getConnectorConfig);
|
|
|
|
|
|
|
|
beforeAll(() => {
|
|
|
|
jest.spyOn(weChatNativeMethods, 'getConfig').mockResolvedValue(mockedConfig);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getAuthorizationUri', () => {
|
|
|
|
afterEach(() => {
|
|
|
|
jest.clearAllMocks();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should get a valid uri by redirectUri and state', async () => {
|
|
|
|
const authorizationUri = await weChatNativeMethods.getAuthorizationUri(
|
|
|
|
'http://localhost:3001/callback',
|
|
|
|
'some_state'
|
|
|
|
);
|
|
|
|
expect(authorizationUri).toEqual(
|
|
|
|
`${authorizationEndpoint}?appid=%3Capp-id%3E&redirect_uri=http%3A%2F%2Flocalhost%3A3001%2Fcallback&scope=snsapi_userinfo&state=some_state`
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getAccessToken', () => {
|
|
|
|
afterEach(() => {
|
|
|
|
nock.cleanAll();
|
|
|
|
jest.clearAllMocks();
|
|
|
|
});
|
|
|
|
|
|
|
|
const accessTokenEndpointUrl = new URL(accessTokenEndpoint);
|
|
|
|
const parameters = new URLSearchParams({
|
|
|
|
appid: '<app-id>',
|
|
|
|
secret: '<app-secret>',
|
|
|
|
code: 'code',
|
|
|
|
grant_type: 'authorization_code',
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should get an accessToken by exchanging with code', async () => {
|
|
|
|
nock(accessTokenEndpointUrl.origin)
|
|
|
|
.get(accessTokenEndpointUrl.pathname)
|
|
|
|
.query(parameters)
|
|
|
|
.reply(200, {
|
|
|
|
access_token: 'access_token',
|
|
|
|
openid: 'openid',
|
|
|
|
});
|
|
|
|
const { accessToken, openid } = await weChatNativeMethods.getAccessToken('code');
|
|
|
|
expect(accessToken).toEqual('access_token');
|
|
|
|
expect(openid).toEqual('openid');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('throws SocialAuthCodeInvalid error if accessToken not found in response', async () => {
|
|
|
|
nock(accessTokenEndpointUrl.origin)
|
|
|
|
.get(accessTokenEndpointUrl.pathname)
|
|
|
|
.query(parameters)
|
|
|
|
.reply(200, {});
|
|
|
|
await expect(weChatNativeMethods.getAccessToken('code')).rejects.toMatchError(
|
|
|
|
new ConnectorError(ConnectorErrorCodes.SocialAuthCodeInvalid)
|
|
|
|
);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('validateConfig', () => {
|
|
|
|
it('should pass on valid config', async () => {
|
|
|
|
await expect(
|
|
|
|
weChatNativeMethods.validateConfig({ appId: 'appId', appSecret: 'appSecret' })
|
|
|
|
).resolves.not.toThrow();
|
|
|
|
});
|
|
|
|
it('should throw on empty config', async () => {
|
|
|
|
await expect(weChatNativeMethods.validateConfig({})).rejects.toThrowError();
|
|
|
|
});
|
|
|
|
it('should throw when missing appSecret', async () => {
|
|
|
|
await expect(weChatNativeMethods.validateConfig({ appId: 'appId' })).rejects.toThrowError();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getUserInfo', () => {
|
|
|
|
afterEach(() => {
|
|
|
|
nock.cleanAll();
|
|
|
|
jest.clearAllMocks();
|
|
|
|
});
|
|
|
|
|
|
|
|
const userInfoEndpointUrl = new URL(userInfoEndpoint);
|
|
|
|
const parameters = new URLSearchParams({ access_token: 'accessToken', openid: 'openid' });
|
|
|
|
|
|
|
|
it('should get valid SocialUserInfo', async () => {
|
|
|
|
nock(userInfoEndpointUrl.origin).get(userInfoEndpointUrl.pathname).query(parameters).reply(0, {
|
|
|
|
unionid: 'this_is_an_arbitrary_wechat_union_id',
|
|
|
|
headimgurl: 'https://github.com/images/error/octocat_happy.gif',
|
|
|
|
nickname: 'wechat bot',
|
|
|
|
});
|
|
|
|
const socialUserInfo = await weChatNativeMethods.getUserInfo({
|
|
|
|
accessToken: 'accessToken',
|
|
|
|
openid: 'openid',
|
|
|
|
});
|
|
|
|
expect(socialUserInfo).toMatchObject({
|
|
|
|
id: 'this_is_an_arbitrary_wechat_union_id',
|
|
|
|
avatar: 'https://github.com/images/error/octocat_happy.gif',
|
|
|
|
name: 'wechat bot',
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('throws SocialAccessTokenInvalid error if errcode is 40001', async () => {
|
|
|
|
nock(userInfoEndpointUrl.origin)
|
|
|
|
.get(userInfoEndpointUrl.pathname)
|
|
|
|
.query(parameters)
|
|
|
|
.reply(200, { errcode: 40_001 });
|
|
|
|
await expect(
|
|
|
|
weChatNativeMethods.getUserInfo({ accessToken: 'accessToken', openid: 'openid' })
|
|
|
|
).rejects.toMatchError(new ConnectorError(ConnectorErrorCodes.SocialAccessTokenInvalid));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('throws unrecognized error', async () => {
|
|
|
|
nock(userInfoEndpointUrl.origin).get(userInfoEndpointUrl.pathname).query(parameters).reply(500);
|
|
|
|
await expect(
|
|
|
|
weChatNativeMethods.getUserInfo({ accessToken: 'accessToken', openid: 'openid' })
|
|
|
|
).rejects.toThrow();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('throws Error if request failed and errcode is not 40001', async () => {
|
|
|
|
nock(userInfoEndpointUrl.origin)
|
|
|
|
.get(userInfoEndpointUrl.pathname)
|
|
|
|
.query(parameters)
|
|
|
|
.reply(200, { errcode: 40_003, errmsg: 'invalid openid' });
|
|
|
|
await expect(
|
|
|
|
weChatNativeMethods.getUserInfo({ accessToken: 'accessToken', openid: 'openid' })
|
|
|
|
).rejects.toMatchError(new Error('invalid openid'));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('throws SocialAccessTokenInvalid error if response code is 401', async () => {
|
|
|
|
nock(userInfoEndpointUrl.origin)
|
|
|
|
.get(userInfoEndpointUrl.pathname)
|
|
|
|
.query(new URLSearchParams({ access_token: 'accessToken' }))
|
|
|
|
.reply(401);
|
|
|
|
await expect(
|
|
|
|
weChatNativeMethods.getUserInfo({ accessToken: 'accessToken' })
|
|
|
|
).rejects.toMatchError(new ConnectorError(ConnectorErrorCodes.SocialAccessTokenInvalid));
|
|
|
|
});
|
|
|
|
});
|