Skip to content

Commit

Permalink
fix(api-rest): infinite retry when clock skew go back and forth (#12488)
Browse files Browse the repository at this point in the history
  • Loading branch information
AllanZhengYP authored Nov 6, 2023
1 parent 93d463c commit 095efac
Show file tree
Hide file tree
Showing 3 changed files with 99 additions and 61 deletions.
152 changes: 95 additions & 57 deletions packages/api-rest/__tests__/RestAPI-test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -475,7 +475,7 @@ describe('Rest API test', () => {
expect(spyon4).toBeCalled();
});

test('clock skew', async () => {
describe('clock skew', () => {
const api = new API({});
api.configure({
endpoints: [
Expand All @@ -486,78 +486,116 @@ describe('Rest API test', () => {
],
});

const normalError = new Error('Response Error');
const signSpy = jest.spyOn(RestClient.prototype as any, '_sign');
signSpy.mockImplementation(() => ({
headers: {
'x-amz-date': DateUtils.getHeaderStringFromDate(
DateUtils.getDateWithClockOffset()
),
Authorization: 'signed',
},
}));

const now = new Date();
// Server is always "correct"
const serverDate = new Date();
const requestDate = new Date();

let serverDate = new Date(now.getTime() + 60 * 60 * 1000);
// Local machine is behind by 1 hour
// It's important to change the _server_ time in this test,
// because the local time "looks correct" to the user & DateUtils
// compares the server response to local time.
serverDate.setHours(serverDate.getHours() + 1);

const clockSkewError: any = new Error('BadRequestException');
const init = {
headers: {
'x-amz-date': DateUtils.getHeaderStringFromDate(requestDate),
},
const clientDate = now;

const clockSkewErrorMockImplementation = async (req: any) => {
const requestDate = DateUtils.getDateFromHeaderString(
req.headers['x-amz-date']
);

const FIVE_MINUTES_IN_MS = 1000 * 60 * 5;
if (
Math.abs(serverDate.getTime() - requestDate.getTime()) >
FIVE_MINUTES_IN_MS
) {
const clockSkewError: any = new Error('BadRequestException');

clockSkewError.response = {
headers: {
'x-amzn-errortype': 'BadRequestException',
date: serverDate.toString(),
},
};
throw clockSkewError;
} else {
return {
data: [{ name: 'Bob' }],
};
}
};

clockSkewError.response = {
headers: {
'x-amzn-errortype': 'BadRequestException',
date: serverDate.toString(),
},
};
beforeEach(() => {
jest.clearAllMocks();
jest
.spyOn(RestClient.prototype as any, 'endpoint')
.mockImplementation(() => 'endpoint');

// Clock should not be skewed yet
expect(DateUtils.getClockOffset()).toBe(0);
// Ensure the errors are the correct type for gating
expect(DateUtils.isClockSkewError(normalError)).toBe(false);
expect(DateUtils.isClockSkewError(clockSkewError)).toBe(true);
expect(DateUtils.isClockSkewed(serverDate)).toBe(true);

jest
.spyOn(RestClient.prototype as any, 'endpoint')
.mockImplementation(() => 'endpoint');

jest.spyOn(Credentials, 'get').mockResolvedValue('creds');
jest.spyOn(RestClient.prototype as any, '_sign').mockReturnValue({
...init,
headers: { ...init.headers, Authorization: 'signed' },
});
mockAxios.mockImplementationOnce(() => {
return new Promise((_, rej) => {
rej(normalError);
});
jest.spyOn(Credentials, 'get').mockResolvedValue('creds');
DateUtils.setClockOffset(0);
});

await expect(api.post('url', 'path', init)).rejects.toThrow(normalError);

// Clock should not be skewed from normal errors
expect(DateUtils.getClockOffset()).toBe(0);

// mock clock skew error response and successful response after retry
mockAxios
.mockImplementationOnce(() => {
test('should not apply offset for normal errors', async () => {
const normalError = new Error('Response Error');
// Clock should not be skewed yet
expect(DateUtils.getClockOffset()).toBe(0);

const signSpy = jest.spyOn(RestClient.prototype as any, '_sign');
signSpy.mockReturnValue({
// ...init,
headers: {
'x-amz-date': DateUtils.getHeaderStringFromDate(clientDate),
Authorization: 'signed',
},
});
mockAxios.mockImplementationOnce(() => {
return new Promise((_, rej) => {
rej(clockSkewError);
rej(normalError);
});
})
.mockResolvedValue({
data: [{ name: 'Bob' }],
});

await expect(api.post('url', 'path', init)).resolves.toEqual([
{ name: 'Bob' },
]);
await expect(api.post('url', 'path', {})).rejects.toThrow(normalError);
expect(mockAxios).toBeCalledTimes(1);
});

// With a clock skew error, the clock will get offset with the difference
expect(DateUtils.getClockOffset()).toBe(
serverDate.getTime() - requestDate.getTime()
);
test('should apply clock skew offset if client clock is skewed', async () => {
mockAxios.mockImplementation(clockSkewErrorMockImplementation);

await expect(api.post('url', 'path', {})).resolves.toEqual([
{ name: 'Bob' },
]);

// With a clock skew error, the clock will get offset with the difference
expect(DateUtils.getClockOffset()).toBe(
serverDate.getTime() - clientDate.getTime()
);
expect(mockAxios).toBeCalledTimes(2);
});

test('should update clock skew offset when client clock is back and forth', async () => {
mockAxios.mockImplementation(clockSkewErrorMockImplementation);
await expect(api.post('url', 'path', {})).resolves.toEqual([
{ name: 'Bob' },
]);

// clock skew of 1 hour should be applied
expect(DateUtils.getClockOffset()).toBe(
serverDate.getTime() - clientDate.getTime()
);

// Mock a consequent clock skew error when client and server clock are in sync but offset is still applied,
// We cannot change client clock, so changing server clock is equivalent to change client clock.
serverDate = now;
await expect(api.post('url', 'path', {})).resolves.toEqual([
{ name: 'Bob' },
]);
});
});

test('cancel request', async () => {
Expand Down
2 changes: 1 addition & 1 deletion packages/api-rest/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -59,7 +59,7 @@
"name": "API (rest client)",
"path": "./lib-esm/index.js",
"import": "{ Amplify, RestAPI }",
"limit": "36665 B"
"limit": "36.68 kB"
}
],
"jest": {
Expand Down
6 changes: 3 additions & 3 deletions packages/api-rest/src/RestClient.ts
Original file line number Diff line number Diff line change
Expand Up @@ -198,9 +198,9 @@ export class RestClient {

// Compare local clock to the server clock
if (DateUtils.isClockSkewed(responseDate)) {
DateUtils.setClockOffset(
responseDate.getTime() - requestDate.getTime()
);
const rawClientDate =
requestDate.getTime() - DateUtils.getClockOffset();
DateUtils.setClockOffset(responseDate.getTime() - rawClientDate);

return this.ajax(urlOrApiInfo, method, init);
}
Expand Down

0 comments on commit 095efac

Please sign in to comment.