nxapi/src/common/auth/nooklink.ts
2022-09-08 19:44:15 +01:00

137 lines
5.1 KiB
TypeScript

import createDebug from 'debug';
import persist from 'node-persist';
import { getToken, Login } from './coral.js';
import NooklinkApi, { NooklinkAuthData, NooklinkUserApi, NooklinkUserAuthData } from '../../api/nooklink.js';
import { Users } from '../../api/nooklink-types.js';
import { checkUseLimit, SHOULD_LIMIT_USE } from './util.js';
import { Jwt } from '../../util/jwt.js';
import { NintendoAccountSessionTokenJwtPayload } from '../../api/na.js';
const debug = createDebug('nxapi:auth:nooklink');
export interface SavedToken extends NooklinkAuthData {}
export async function getWebServiceToken(
storage: persist.LocalStorage, token: string, proxy_url?: string,
allow_fetch_token = false, ratelimit = SHOULD_LIMIT_USE
) {
if (!token) {
console.error('No token set. Set a Nintendo Account session token using the `--token` option or by running `nxapi nso token`.');
throw new Error('Invalid token');
}
const existingToken: SavedToken | undefined = await storage.getItem('NookToken.' + token);
if (!existingToken || existingToken.expires_at <= Date.now()) {
if (!allow_fetch_token) {
throw new Error('No valid NookLink web service token');
}
const { default: { coral_gws_nooklink: config } } = await import('../remote-config.js');
if (!config) throw new Error('Remote configuration prevents NookLink authentication');
if (ratelimit) {
const [jwt, sig] = Jwt.decode<NintendoAccountSessionTokenJwtPayload>(token);
await checkUseLimit(storage, 'nooklink', jwt.payload.sub);
}
console.warn('Authenticating to NookLink');
debug('Authenticating to NookLink');
const {nso, data} = await getToken(storage, token, proxy_url);
if (data[Login]) {
const announcements = await nso.getAnnouncements();
const friends = await nso.getFriendList();
const webservices = await nso.getWebServices();
const activeevent = await nso.getActiveEvent();
}
const existingToken: SavedToken = await NooklinkApi.loginWithCoral(nso, data.user);
await storage.setItem('NookToken.' + token, existingToken);
return {
nooklink: NooklinkApi.createWithSavedToken(existingToken),
data: existingToken,
};
}
debug('Using existing web service token');
return {
nooklink: NooklinkApi.createWithSavedToken(existingToken),
data: existingToken,
};
}
export interface SavedUserToken extends NooklinkUserAuthData {}
type PromiseValue<T> = T extends PromiseLike<infer R> ? R : never;
export async function getUserToken(
storage: persist.LocalStorage, nintendoAccountToken: string, user?: string,
proxy_url?: string, allow_fetch_token = false, ratelimit = SHOULD_LIMIT_USE
) {
let wst: PromiseValue<ReturnType<typeof getWebServiceToken>> | null = null;
if (!user) {
let cachedUsers: {
users: Users;
expires_at: number;
} | undefined = await storage.getItem('NookUsers.' + nintendoAccountToken);
if (!cachedUsers || cachedUsers.expires_at <= Date.now()) {
if (!wst) wst = await getWebServiceToken(storage, nintendoAccountToken, proxy_url, allow_fetch_token);
const {nooklink, data: webserviceToken} = wst;
debug('Fetching users');
const users = await nooklink.getUsers();
await storage.setItem('NookUsers.' + nintendoAccountToken, cachedUsers = {
users,
expires_at: webserviceToken.expires_at,
});
}
if (!cachedUsers.users.users.length) {
throw new Error('No Animal Crossing: New Horizons save data linked to NookLink');
}
if (cachedUsers.users.users.length > 1) {
console.warn('More than 1 NookLink user is linked to this Nintendo Account. The first player will be used. Use `--islander` to set a specific user.');
}
user = cachedUsers.users.users[0].id;
}
const existingToken: SavedUserToken | undefined = await storage.getItem('NookAuthToken.' + nintendoAccountToken + '.' + user);
if (!existingToken || existingToken.token.expireAt <= (Date.now() / 1000)) {
if (!wst) wst = await getWebServiceToken(storage, nintendoAccountToken, proxy_url, allow_fetch_token);
const {nooklink, data: webserviceToken} = wst;
if (ratelimit) {
const [jwt, sig] = Jwt.decode<NintendoAccountSessionTokenJwtPayload>(nintendoAccountToken);
await checkUseLimit(storage, 'nooklink-user', jwt.payload.sub);
}
console.warn('Authenticating to NookLink as user %s', user);
debug('Authenticating to NookLink as user %s', user);
const {nooklinkuser, data} = await nooklink.createUserClient(user);
const existingToken: SavedUserToken = data;
await storage.setItem('NookAuthToken.' + nintendoAccountToken + '.' + user, existingToken);
return {nooklinkuser, data: existingToken};
}
debug('Using existing NookLink auth token');
return {
nooklinkuser: NooklinkUserApi.createWithSavedToken(existingToken),
data: existingToken,
};
}