nxapi/src/common/auth/nooklink.ts
2023-07-12 23:15:37 +01:00

241 lines
8.9 KiB
TypeScript

import persist from 'node-persist';
import { Response } from 'undici';
import { getToken, Login } from './coral.js';
import NooklinkApi, { NooklinkAuthData, NooklinkUserApi, NooklinkUserAuthData } from '../../api/nooklink.js';
import { Users, WebServiceError } from '../../api/nooklink-types.js';
import { checkMembershipActive, checkUseLimit, SHOULD_LIMIT_USE } from './util.js';
import createDebug from '../../util/debug.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');
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();
}
checkMembershipActive(data);
let attempt;
if (ratelimit) {
const [jwt, sig] = Jwt.decode<NintendoAccountSessionTokenJwtPayload>(token);
attempt = await checkUseLimit(storage, 'nooklink', jwt.payload.sub);
}
try {
console.warn('Authenticating to NookLink');
debug('Authenticating to NookLink');
const existingToken: SavedToken = await NooklinkApi.loginWithCoral(nso, data.user);
await storage.setItem('NookToken.' + token, existingToken);
const nooklink = NooklinkApi.createWithSavedToken(existingToken);
nooklink.onTokenExpired = createTokenExpiredHandler(storage, token, nooklink, {
existingToken,
znc_proxy_url: proxy_url,
});
return {nooklink, data: existingToken};
} catch (err) {
await attempt?.recordError(err);
throw err;
}
}
debug('Using existing web service token');
const nooklink = NooklinkApi.createWithSavedToken(existingToken);
nooklink.onTokenExpired = createTokenExpiredHandler(storage, token, nooklink, {
existingToken,
znc_proxy_url: proxy_url,
});
return {nooklink, data: existingToken};
}
function createTokenExpiredHandler(
storage: persist.LocalStorage, token: string, nooklink: NooklinkApi,
renew_token_data: {existingToken: SavedToken; znc_proxy_url?: string},
ratelimit = true
) {
return (data?: WebServiceError, response?: Response) => {
debug('Token expired, renewing', data);
return renewToken(storage, token, nooklink, renew_token_data, ratelimit);
};
}
async function renewToken(
storage: persist.LocalStorage, token: string, nooklink: NooklinkApi,
renew_token_data: {existingToken: SavedToken; znc_proxy_url?: string},
ratelimit = true
) {
if (ratelimit) {
const [jwt, sig] = Jwt.decode<NintendoAccountSessionTokenJwtPayload>(token);
await checkUseLimit(storage, 'nooklink', jwt.payload.sub);
}
const {nso, data} = await getToken(storage, token, renew_token_data.znc_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 nooklink.renewTokenWithCoral(nso, data.user);
await storage.setItem('NookToken.' + token, existingToken);
renew_token_data.existingToken = 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;
nooklinkuser.onTokenExpired = createUserTokenExpiredHandler(storage, nintendoAccountToken, nooklinkuser, {
existingToken,
znc_proxy_url: proxy_url,
nooklink,
});
await storage.setItem('NookAuthToken.' + nintendoAccountToken + '.' + user, existingToken);
return {nooklinkuser, data: existingToken};
}
debug('Using existing NookLink auth token');
const nooklinkuser = NooklinkUserApi.createWithSavedToken(existingToken);
nooklinkuser.onTokenExpired = createUserTokenExpiredHandler(storage, nintendoAccountToken, nooklinkuser, {
existingToken,
znc_proxy_url: proxy_url,
nooklink: null,
});
return {nooklinkuser, data: existingToken};
}
function createUserTokenExpiredHandler(
storage: persist.LocalStorage, token: string, nooklinkuser: NooklinkUserApi,
renew_token_data: {existingToken: SavedUserToken; znc_proxy_url?: string; nooklink: NooklinkApi | null},
ratelimit = true
) {
return (data?: WebServiceError, response?: Response) => {
debug('Token expired', nooklinkuser.user_id, data);
return renewUserToken(storage, token, nooklinkuser, renew_token_data);
};
}
async function renewUserToken(
storage: persist.LocalStorage, token: string, nooklinkuser: NooklinkUserApi,
renew_token_data: {existingToken: SavedUserToken; znc_proxy_url?: string; nooklink: NooklinkApi | null},
ratelimit = true
) {
if (!renew_token_data.nooklink) {
const wst = await getWebServiceToken(storage, token, renew_token_data.znc_proxy_url, true, ratelimit);
const {nooklink, data: webserviceToken} = wst;
renew_token_data.nooklink = nooklink;
}
if (ratelimit) {
const [jwt, sig] = Jwt.decode<NintendoAccountSessionTokenJwtPayload>(token);
await checkUseLimit(storage, 'nooklink-user', jwt.payload.sub);
}
const data = await nooklinkuser.renewToken(renew_token_data.nooklink);
const existingToken: SavedUserToken = {
...renew_token_data.existingToken,
...data,
};
await storage.setItem('NookAuthToken.' + token + '.' + nooklinkuser.user_id, existingToken);
renew_token_data.existingToken = existingToken;
}