Skip to content

Commit

Permalink
Revert "Gateway updates"
Browse files Browse the repository at this point in the history
This reverts commit 2e46226.
  • Loading branch information
thostetler committed Jan 10, 2025
1 parent f9598b8 commit 4c3c631
Show file tree
Hide file tree
Showing 23 changed files with 155 additions and 240 deletions.
2 changes: 1 addition & 1 deletion global.d.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ declare module 'iron-session' {
token?: {
access_token: string;
anonymous: boolean;
expires_at: string;
expire_in: string;
username: string;
};
isAuthenticated?: boolean;
Expand Down
10 changes: 5 additions & 5 deletions src/api/__tests__/api.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,17 +11,17 @@ global.alert = vi.fn();

const API_USER = '/api/user';

const mockUserData: Pick<IBootstrapPayload, 'username' | 'access_token' | 'anonymous' | 'expires_at'> = {
const mockUserData: Pick<IBootstrapPayload, 'username' | 'access_token' | 'anonymous' | 'expire_in'> = {
username: 'anonymous@ads',
access_token: 'foo_access_token',
anonymous: true,
expires_at: '99999999999999999',
expire_in: '99999999999999999',
};
const invalidMockUserData: Pick<IBootstrapPayload, 'username' | 'access_token' | 'anonymous' | 'expires_at'> = {
const invalidMockUserData: Pick<IBootstrapPayload, 'username' | 'access_token' | 'anonymous' | 'expire_in'> = {
username: 'anonymous@ads',
access_token: '',
anonymous: true,
expires_at: '',
expire_in: '',
};

const testHandlerWith200 = rest.get('*test', (_, res, ctx) => {
Expand Down Expand Up @@ -182,7 +182,7 @@ test('passing token initially skips bootstrap', async ({ server }: TestContext)
test('expired userdata causes bootstrap', async ({ server }: TestContext) => {
const { onRequest: onReq } = createServerListenerMocks(server);
server.use(testHandlerWith200);
api.setUserData({ ...mockUserData, expires_at: '999' });
api.setUserData({ ...mockUserData, expire_in: '999' });
await testRequest();

expect(urls(onReq)).toStrictEqual([API_USER, '/test']);
Expand Down
10 changes: 1 addition & 9 deletions src/api/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,7 @@ const resolveApiBaseUrl = (defaultBaseUrl = ''): string => {
}

// use a known URL for development
if (
typeof window !== 'undefined' &&
process.env.NODE_ENV === 'development' &&
typeof process.env.NEXT_PUBLIC_API_HOST_CLIENT === 'string'
) {
if (process.env.NODE_ENV === 'development' && typeof process.env.NEXT_PUBLIC_API_HOST_CLIENT === 'string') {
return process.env.NEXT_PUBLIC_API_HOST_CLIENT;
}

Expand All @@ -35,10 +31,6 @@ const resolveApiBaseUrl = (defaultBaseUrl = ''): string => {
export const defaultRequestConfig: AxiosRequestConfig = {
baseURL: resolveApiBaseUrl(),
withCredentials: true,
headers: {
'Content-Type': 'application/json',
},

timeout: typeof window === 'undefined' ? APP_DEFAULTS.SSR_API_TIMEOUT : APP_DEFAULTS.API_TIMEOUT,
paramsSerializer: {
serialize: (params) =>
Expand Down
113 changes: 48 additions & 65 deletions src/api/models.ts
Original file line number Diff line number Diff line change
Expand Up @@ -190,84 +190,67 @@ export const biblibSorts = [
] as const;

export enum ApiTargets {
// Accounts
BOOTSTRAP = '/accounts/bootstrap',
CHANGE_EMAIL = '/accounts/user/change-email',
CHANGE_PASSWORD = '/accounts/user/change-password',
SEARCH = '/search/query',
QTREE = '/search/qtree',
BIGQUERY = '/search/bigquery',
EXPORT = '/export',
SERVICE_AUTHOR_NETWORK = '/vis/author-network',
SERVICE_PAPER_NETWORK = '/vis/paper-network',
SERVICE_WORDCLOUD = '/vis/word-cloud',
SERVICE_METRICS = '/metrics',
SERVICE_OBJECTS = '/objects',
SERVICE_OBJECTS_QUERY = '/objects/query',
SERVICE_CITATION_HELPER = '/citation_helper',
SERVICE_AUTHOR_AFFILIATION_EXPORT = '/authoraff',
MYADS_STORAGE = '/vault',
MYADS_STORAGE_QUERY = '/vault/query',
MYADS_NOTIFICATIONS = '/vault/notifications',
MYADS_NOTIFICATIONS_QUERY = '/vault/notification_query',
LINK_SERVERS = '/vault/configuration/link_servers',
AUTHOR_AFFILIATION_SEARCH = '/author-affiliation/search',
AUTHOR_AFFILIATION_EXPORT = '/author-affiliation/export',
RESOLVER = '/resolver',
CSRF = '/accounts/csrf',
INFO = '/accounts/info',
LOGIN = '/accounts/user/login',
LOGOUT = '/accounts/user/logout',
PROTECTED = '/accounts/protected',
// REGISTER = '/accounts/register',
RESET_PASSWORD = '/accounts/reset-password',
STATUS = '/accounts/status',
TOKEN = '/accounts/user/token',
USER = '/accounts/user',
USER_DATA = '/vault/user-data',
SITE_CONFIGURATION = '/vault/configuration',
TOKEN = '/accounts/token',
INFO = '/accounts/info',
LOGIN = '/accounts/login',
LOGOUT = '/accounts/logout',
REGISTER = '/accounts/register',
VERIFY = '/accounts/verify',
DELETE = '/accounts/user/delete',

// Author Affiliation
AUTHOR_AFFILIATION_EXPORT = '/author-affiliation/export',
AUTHOR_AFFILIATION_SEARCH = '/author-affiliation/search',

// Library Operations
DOCUMENTS = '/biblib/documents',
LIBRARIES = '/biblib/libraries',
LIBRARY_IMPORT_ADS2_AUTH = '/harbour/auth/twopointoh',
LIBRARY_IMPORT_ADS2_TO_BBB = '/biblib/twopointoh',
RESET_PASSWORD = '/accounts/reset-password',
CHANGE_PASSWORD = '/accounts/change-password',
CHANGE_EMAIL = '/accounts/change-email',
RECOMMENDER = '/recommender',
GRAPHICS = '/graphics',
FEEDBACK = '/feedback/userfeedback',
LIBRARY_IMPORT_CLASSIC_AUTH = '/harbour/auth/classic',
LIBRARY_IMPORT_CLASSIC_MIRRORS = '/harbour/mirrors',
LIBRARY_IMPORT_CLASSIC_TO_BBB = '/biblib/classic',
LIBRARY_IMPORT_CREDENTIALS = '/harbour/user',
LIBRARY_IMPORT_MENDELEY = '/harbour/export/twopointoh/mendeley',
LIBRARY_IMPORT_ADS2_AUTH = '/harbour/auth/twopointoh',
LIBRARY_IMPORT_ADS2_TO_BBB = '/biblib/twopointoh',
LIBRARY_IMPORT_ZOTERO = '/harbour/export/twopointoh/zotero',
LIBRARY_NOTES = 'biblib/notes',
LIBRARY_OPERATION = 'biblib/libraries/operations',
LIBRARY_QUERY = 'biblib/query',
LIBRARY_TRANSFER = '/biblib/transfer',
PERMISSIONS = '/biblib/permissions',

// Orcid Operations
LIBRARY_IMPORT_MENDELEY = '/harbour/export/twopointoh/mendeley',
LIBRARY_IMPORT_CREDENTIALS = '/harbour/user',
ORCID_PREFERENCES = '/orcid/preferences',
ORCID = '/orcid',
ORCID_EXCHANGE_TOKEN = '/orcid/exchangeOAuthCode',
ORCID_NAME = '/orcid/orcid-name',
ORCID_PREFERENCES = '/orcid/preferences',
ORCID_PROFILE = 'orcid-profile',
ORCID_WORK = 'orcid-work',
ORCID_WORKS = 'orcid-works',

// Vault Operations
MYADS_NOTIFICATIONS = '/vault/notifications',
MYADS_NOTIFICATIONS_QUERY = '/vault/notification_query',
MYADS_STORAGE = '/vault',
MYADS_STORAGE_QUERY = '/vault/query',
SITE_CONFIGURATION = '/vault/configuration',
USER_DATA = '/vault/user-data',
LINK_SERVERS = '/vault/configuration/link_servers',

// Search and Query
BIGQUERY = '/search/bigquery',
QTREE = '/search/qtree',
SEARCH = '/search/query',

// Services
SERVICE_AUTHOR_AFFILIATION_EXPORT = '/authoraff',
SERVICE_AUTHOR_NETWORK = '/vis/author-network',
SERVICE_CITATION_HELPER = '/citation_helper',
SERVICE_METRICS = '/metrics',
SERVICE_OBJECTS = '/objects',
SERVICE_OBJECTS_QUERY = '/objects/query',
SERVICE_PAPER_NETWORK = '/vis/paper-network',
SERVICE_WORDCLOUD = '/vis/word-cloud',

// Miscellaneous
EXPORT = '/export',
FEEDBACK = '/feedback',
GRAPHICS = '/graphics',
RECOMMENDER = '/recommender',
ORCID_WORK = 'orcid-work',
ORCID_PROFILE = 'orcid-profile',
ORCID_EXCHANGE_TOKEN = '/orcid/exchangeOAuthCode',
LIBRARIES = '/biblib/libraries',
LIBRARY_TRANSFER = '/biblib/transfer',
LIBRARY_OPERATION = 'biblib/libraries/operations',
LIBRARY_QUERY = 'biblib/query',
LIBRARY_NOTES = 'biblib/notes',
DOCUMENTS = '/biblib/documents',
PERMISSIONS = '/biblib/permissions',
REFERENCE = '/reference/text',
RESOLVER = '/resolver',
JOURNAL = 'journals/journal',
JOURNAL_SUMMARY = 'journals/browse',
JOURNAL_ISSN = 'journals/issn',
Expand Down
1 change: 0 additions & 1 deletion src/api/objects/objects.ts
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,6 @@ export const resolveObjectQuerySSR = async (params: IObjectsQueryApiParams, ctx:
method: 'POST',
data: { query: [query] },
headers: {
...defaultRequestConfig.headers,
...pick(TRACING_HEADERS, ctx.req.headers),
Authorization: `Bearer ${token}`,
},
Expand Down
1 change: 0 additions & 1 deletion src/api/search/search.ts
Original file line number Diff line number Diff line change
Expand Up @@ -449,7 +449,6 @@ export const fetchSearchSSR = async (
params: finalParams,
signal,
headers: {
...defaultRequestConfig.headers,
Authorization: `Bearer ${token}`,
...pick(TRACING_HEADERS, ctx.req.headers),
},
Expand Down
8 changes: 2 additions & 6 deletions src/api/user/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -41,14 +41,12 @@ export interface IBootstrapPayload {
ratelimit: number;
anonymous: boolean;
client_secret: string;
expires_at: string;
expire_in: string;
refresh_token: string;
given_name: string;
family_name: string;
message?: string;
}

export type IUserData = Pick<IBootstrapPayload, 'username' | 'anonymous' | 'access_token' | 'expires_at'>;
export type IUserData = Pick<IBootstrapPayload, 'username' | 'anonymous' | 'access_token' | 'expire_in'>;

export interface IUserForgotPasswordCredentials {
email: string;
Expand All @@ -61,8 +59,6 @@ export interface IUserCredentials {
}

export interface IUserRegistrationCredentials {
givenName: string;
familyName: string;
email: string;
password: string;
confirmPassword: string;
Expand Down
8 changes: 3 additions & 5 deletions src/api/user/user.ts
Original file line number Diff line number Diff line change
Expand Up @@ -90,10 +90,8 @@ const registerUser: MutationFunction<IBasicAccountsResponse, IUserRegistrationCr
const config = await configWithCSRF({
...defaultRequestConfig,
method: 'POST',
url: ApiTargets.USER,
url: ApiTargets.REGISTER,
data: {
given_name: credentials.givenName,
family_name: credentials.familyName,
email: credentials.email,
password1: credentials.password,
password2: credentials.confirmPassword,
Expand Down Expand Up @@ -210,8 +208,8 @@ const resetUserPassword: MutationFunction<IBasicAccountsResponse, IUserResetPass
export const deleteUserAccount: MutationFunction<IBasicAccountsResponse, unknown> = async () => {
const config = await configWithCSRF({
...defaultRequestConfig,
method: 'DELETE',
url: ApiTargets.USER,
method: 'POST',
url: ApiTargets.DELETE,
});

const { data } = await api.request<IBasicAccountsResponse>(config);
Expand Down
47 changes: 18 additions & 29 deletions src/auth-utils.ts
Original file line number Diff line number Diff line change
@@ -1,22 +1,19 @@
import axios, { AxiosRequestConfig, AxiosResponse } from 'axios';
import { defaultRequestConfig } from '@/api/config';

import { defaultRequestConfig } from '@/api/config';
import { isNil } from 'ramda';
import { isPast, parseISO } from 'date-fns';
import { APP_DEFAULTS } from '@/config';
import { IBootstrapPayload, ICSRFResponse, IUserData } from '@/api/user/types';
import { ApiTargets } from '@/api/models';
import { ApiRequestConfig } from '@/api/api';
import { edgeLogger as logger } from '@/logger';

const fetchCSRF = async () => {
const config: AxiosRequestConfig = {
const fetchCSRF = async () =>
await axios.get<ICSRFResponse, AxiosResponse<ICSRFResponse>>(ApiTargets.CSRF, {
...defaultRequestConfig,
url: ApiTargets.CSRF,
timeout: APP_DEFAULTS.API_TIMEOUT,
};
logger.debug({ config }, 'Fetching CSRF token');
return axios.request<ICSRFResponse, AxiosResponse<ICSRFResponse>>(config);
};
});

export const configWithCSRF = async (config: ApiRequestConfig): Promise<ApiRequestConfig> => {
const csrfRes = await fetchCSRF();
Expand Down Expand Up @@ -69,31 +66,23 @@ export const hash = async (str?: string) => {
* Checks if the user data is valid
* @param userData
*/
export const isUserData = (userData?: IUserData): userData is IUserData =>
!isNil(userData) &&
typeof userData.access_token === 'string' &&
typeof userData.expires_at === 'string' &&
userData.access_token.length > 0 &&
userData.expires_at.length > 0;

/**
* Checks if a token is expired based on the expiration time.
*
* @param {string} expiresAt - The expiration time of the token in seconds since the Unix epoch.
* @returns {boolean} - Returns true if the current time is greater than or equal to the expiration time, false otherwise.
*/
export const isTokenExpired = (expiresAt: string): boolean => {
const currentTime = Math.floor(Date.now() / 1000);
const tokenExpiryTime = parseInt(expiresAt, 10);
return currentTime >= tokenExpiryTime;
export const isUserData = (userData?: IUserData): userData is IUserData => {
return (
!isNil(userData) &&
typeof userData.access_token === 'string' &&
typeof userData.expire_in === 'string' &&
userData.access_token.length > 0 &&
userData.expire_in.length > 0
);
};

/**
* Checks if the token is valid
* Checks if the user data is valid and the token is not expired
* @param userData
*/
export const isValidToken = (userData?: IUserData): boolean =>
isUserData(userData) && !isTokenExpired(userData.expires_at);
export const isValidToken = (userData?: IUserData): boolean => {
return isUserData(userData) && !isPast(parseISO(userData.expire_in));
};

/**
* Checks if the user is authenticated
Expand All @@ -112,7 +101,7 @@ export const pickUserData = (userData?: IUserData | IBootstrapPayload) => {
}
return {
access_token: userData.access_token,
expires_at: userData.expires_at,
expire_in: userData.expire_in,
username: userData.username,
anonymous: userData.anonymous,
};
Expand Down
2 changes: 1 addition & 1 deletion src/components/__mocks__/session.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ export const mockSession: IronSessionData = {
token: {
access_token: '',
anonymous: true,
expires_at: '1950000000',
expire_in: '9999-01-01T00:00:00',
username: 'anonymous',
},
isAuthenticated: false,
Expand Down
3 changes: 1 addition & 2 deletions src/lib/useSession.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,10 @@
import api from '@/api/api';
import axios from 'axios';
import { useEffect } from 'react';
import { useUser } from '@/lib/useUser';
import { useMutation } from '@tanstack/react-query';
import { ILogoutResponse } from '@/pages/api/auth/logout';
import { useRouter } from 'next/router';
import { isAuthenticated } from '@/auth-utils';
import api, { isAuthenticated } from '@/api/api';

/**
* Provides access to the user session and methods to logout
Expand Down
2 changes: 1 addition & 1 deletion src/middlewares/botCheck.ts
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ const crawlerCheck = async (req: NextRequest, ip: string, ua: string) => {

const baseToken: IronSessionData['token'] = {
anonymous: true,
expires_at: '99999999999999',
expire_in: '9999-01-01T00:00:00',
username: 'anonymous',
access_token: 'no-token',
};
Expand Down
Loading

0 comments on commit 4c3c631

Please sign in to comment.