Skip to content

Commit

Permalink
Removes irrelevant upsell in the timeout message (#1599) (#1653)
Browse files Browse the repository at this point in the history
Resolves #1594

Signed-off-by: Tommy Markley <[email protected]>
  • Loading branch information
Tommy Markley authored Jun 2, 2022
1 parent 81d130d commit cada6a1
Show file tree
Hide file tree
Showing 6 changed files with 11 additions and 134 deletions.
1 change: 0 additions & 1 deletion .lycheeexclude
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,6 @@ http://evil.com/
https://opensearch.org/cool/path
https://opensearch.org/redirect
http://www.opensearch.org/painlessDocs
https://opensearch.org/subscriptions
https://www.hostedgraphite.com/

# External urls
Expand Down
1 change: 0 additions & 1 deletion src/plugins/data/public/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -389,7 +389,6 @@ export {
// errors
SearchError,
SearchTimeoutError,
TimeoutErrorMode,
PainlessError,
} from './search';

Expand Down
18 changes: 1 addition & 17 deletions src/plugins/data/public/public.api.md
Original file line number Diff line number Diff line change
Expand Up @@ -2014,8 +2014,6 @@ export class SearchInterceptor {
// (undocumented)
protected readonly deps: SearchInterceptorDeps;
// (undocumented)
protected getTimeoutMode(): TimeoutErrorMode;
// (undocumented)
protected handleSearchError(e: any, request: IOpenSearchDashboardsSearchRequest, timeoutSignal: AbortSignal, appAbortSignal?: AbortSignal): Error;
// @internal
protected pendingCount$: BehaviorSubject<number>;
Expand Down Expand Up @@ -2150,11 +2148,9 @@ export interface SearchSourceFields {
//
// @public
export class SearchTimeoutError extends OsdError {
constructor(err: Error, mode: TimeoutErrorMode);
constructor(err: Error);
// (undocumented)
getErrorMessage(application: ApplicationStart): JSX.Element;
// (undocumented)
mode: TimeoutErrorMode;
}

// Warning: (ae-missing-release-tag) "SortDirection" is exported by the package, but it is missing a release tag (@alpha, @beta, @public, or @internal)
Expand Down Expand Up @@ -2229,18 +2225,6 @@ export class TimeHistory {
// @public (undocumented)
export type TimeHistoryContract = PublicMethodsOf<TimeHistory>;

// Warning: (ae-missing-release-tag) "TimeoutErrorMode" is exported by the package, but it is missing a release tag (@alpha, @beta, @public, or @internal)
//
// @public (undocumented)
export enum TimeoutErrorMode {
// (undocumented)
CHANGE = 2,
// (undocumented)
CONTACT = 1,
// (undocumented)
UPGRADE = 0
}

// Warning: (ae-missing-release-tag) "TimeRange" is exported by the package, but it is missing a release tag (@alpha, @beta, @public, or @internal)
//
// @public (undocumented)
Expand Down
31 changes: 4 additions & 27 deletions src/plugins/data/public/search/errors/timeout_error.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@
* GitHub history for details.
*/

import { SearchTimeoutError, TimeoutErrorMode } from './timeout_error';
import { SearchTimeoutError, TIMEOUT_MESSAGE } from './timeout_error';

import { coreMock } from '../../../../../core/public/mocks';
const startMock = coreMock.createStart();
Expand All @@ -44,32 +44,9 @@ describe('SearchTimeoutError', () => {
startMock.application.navigateToApp.mockImplementation(jest.fn());
});

it('Should navigate to upgrade', () => {
const e = new SearchTimeoutError(new AbortError(), TimeoutErrorMode.UPGRADE);
it('Should create timeout message', () => {
const e = new SearchTimeoutError(new AbortError());
const component = mount(e.getErrorMessage(startMock.application));

expect(component.find('EuiButton').length).toBe(1);
component.find('EuiButton').simulate('click');
expect(startMock.application.navigateToUrl).toHaveBeenCalledWith(
'https://opensearch.org/subscriptions'
);
});

it('Should create contact admin message', () => {
const e = new SearchTimeoutError(new AbortError(), TimeoutErrorMode.CONTACT);
const component = mount(e.getErrorMessage(startMock.application));

expect(component.find('EuiButton').length).toBe(0);
});

it('Should navigate to settings', () => {
const e = new SearchTimeoutError(new AbortError(), TimeoutErrorMode.CHANGE);
const component = mount(e.getErrorMessage(startMock.application));

expect(component.find('EuiButton').length).toBe(1);
component.find('EuiButton').simulate('click');
expect(startMock.application.navigateToApp).toHaveBeenCalledWith('management', {
path: '/opensearch-dashboards/settings',
});
expect(component.html()).toEqual(TIMEOUT_MESSAGE);
});
});
81 changes: 4 additions & 77 deletions src/plugins/data/public/search/errors/timeout_error.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -32,96 +32,23 @@

import React from 'react';
import { i18n } from '@osd/i18n';
import { EuiButton, EuiSpacer, EuiText } from '@elastic/eui';
import { ApplicationStart } from 'opensearch-dashboards/public';
import { OsdError } from '../../../../opensearch_dashboards_utils/common';

export enum TimeoutErrorMode {
UPGRADE,
CONTACT,
CHANGE,
}

export const TIMEOUT_MESSAGE: string =
'Your query has timed out. Contact your system administrator to review your index strategy or increase the run time.';
/**
* Request Failure - When an entire multi request fails
* @param {Error} err - the Error that came back
*/
export class SearchTimeoutError extends OsdError {
public mode: TimeoutErrorMode;
constructor(err: Error, mode: TimeoutErrorMode) {
constructor(err: Error) {
super(`Request timeout: ${JSON.stringify(err?.message)}`);
this.mode = mode;
}

private getMessage() {
switch (this.mode) {
case TimeoutErrorMode.UPGRADE:
return i18n.translate('data.search.upgradeLicense', {
defaultMessage:
'Your query has timed out. With our free Basic tier, your queries never time out.',
});
case TimeoutErrorMode.CONTACT:
return i18n.translate('data.search.timeoutContactAdmin', {
defaultMessage:
'Your query has timed out. Contact your system administrator to increase the run time.',
});
case TimeoutErrorMode.CHANGE:
return i18n.translate('data.search.timeoutIncreaseSetting', {
defaultMessage:
'Your query has timed out. Increase run time with the search timeout advanced setting.',
});
}
}

private getActionText() {
switch (this.mode) {
case TimeoutErrorMode.UPGRADE:
return i18n.translate('data.search.upgradeLicenseActionText', {
defaultMessage: 'Upgrade now',
});
break;
case TimeoutErrorMode.CHANGE:
return i18n.translate('data.search.timeoutIncreaseSettingActionText', {
defaultMessage: 'Edit setting',
});
break;
}
}

private onClick(application: ApplicationStart) {
switch (this.mode) {
case TimeoutErrorMode.UPGRADE:
application.navigateToUrl('https://opensearch.org/subscriptions');
break;
case TimeoutErrorMode.CHANGE:
application.navigateToApp('management', {
path: `/opensearch-dashboards/settings`,
});
break;
}
}

public getErrorMessage(application: ApplicationStart) {
const actionText = this.getActionText();
return (
<>
{this.getMessage()}
{actionText && (
<>
<EuiSpacer size="s" />
<EuiText textAlign="right">
<EuiButton
color="danger"
onClick={() => this.onClick(application)}
size="s"
data-test-subj="searchTimeoutError"
>
{actionText}
</EuiButton>
</EuiText>
</>
)}
</>
<>{i18n.translate('data.search.timeoutContactAdmin', { defaultMessage: TIMEOUT_MESSAGE })}</>
);
}
}
13 changes: 2 additions & 11 deletions src/plugins/data/public/search/search_interceptor.ts
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ import {
OPENSEARCH_SEARCH_STRATEGY,
} from '../../common';
import { SearchUsageCollector } from './collectors';
import { SearchTimeoutError, PainlessError, isPainlessError, TimeoutErrorMode } from './errors';
import { SearchTimeoutError, PainlessError, isPainlessError } from './errors';
import { toMountPoint } from '../../../opensearch_dashboards_react/public';

export interface SearchInterceptorDeps {
Expand Down Expand Up @@ -83,14 +83,6 @@ export class SearchInterceptor {
});
}

/*
* @returns `TimeoutErrorMode` indicating what action should be taken in case of a request timeout based on license and permissions.
* @internal
*/
protected getTimeoutMode() {
return TimeoutErrorMode.UPGRADE;
}

/*
* @returns `Error` a search service specific error or the original error, if a specific error can't be recognized.
* @internal
Expand All @@ -103,8 +95,7 @@ export class SearchInterceptor {
): Error {
if (timeoutSignal.aborted || get(e, 'body.message') === 'Request timed out') {
// Handle a client or a server side timeout
const err = new SearchTimeoutError(e, this.getTimeoutMode());

const err = new SearchTimeoutError(e);
// Show the timeout error here, so that it's shown regardless of how an application chooses to handle errors.
this.showTimeoutError(err);
return err;
Expand Down

0 comments on commit cada6a1

Please sign in to comment.