Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: ACNA-2890 - fix error handling in library #144

Merged
merged 3 commits into from
Mar 26, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
83 changes: 48 additions & 35 deletions lib/AdobeState.js
Original file line number Diff line number Diff line change
Expand Up @@ -72,39 +72,44 @@ async function _wrap (promise, params) {
try {
response = await promise
logger.debug('response', response)
// reuse code in exception handler, for any other network exceptions
if (!response.ok) {
// no exception on 404
if (response.status === 404) {
return null
} else {
const e = new Error(response.statusText)
e.status = response.status
e.internal = response
throw e
}
}
} catch (e) {
const status = e.status || e.code
const copyParams = cloneDeep(params)
logger.debug(`got internal error with status ${status}: ${e.message} `)
switch (status) {
case 401:
return logAndThrow(new codes.ERROR_UNAUTHORIZED({ messageValues: ['underlying DB provider'], sdkDetails: copyParams }))
case 403:
return logAndThrow(new codes.ERROR_BAD_CREDENTIALS({ messageValues: ['underlying DB provider'], sdkDetails: copyParams }))
case 413:
return logAndThrow(new codes.ERROR_PAYLOAD_TOO_LARGE({ messageValues: ['underlying DB provider'], sdkDetails: copyParams }))
case 429:
return logAndThrow(new codes.ERROR_REQUEST_RATE_TOO_HIGH({ sdkDetails: copyParams }))
default:
// NOTE: we should throw a different error if its not a response error
return logAndThrow(new codes.ERROR_INTERNAL({ messageValues: [`unexpected response from provider with status: ${status}`], sdkDetails: { ...cloneDeep(params), _internal: e.internal || e.message } }))
}
logAndThrow(e)
}

handleResponse(response, params)
return response
}

/**
* Handle a network response.
*
* @param {Response} response a fetch Response
* @param {object} params the params to the network call
* @returns {void}
*/
function handleResponse (response, params) {
if (response.ok) {
return
}

const copyParams = cloneDeep(params)
switch (response.status) {
case 404:
// do nothing, no exception on 404
break
case 401:
return logAndThrow(new codes.ERROR_UNAUTHORIZED({ messageValues: ['underlying DB provider'], sdkDetails: copyParams }))
case 403:
return logAndThrow(new codes.ERROR_BAD_CREDENTIALS({ messageValues: ['underlying DB provider'], sdkDetails: copyParams }))
case 413:
return logAndThrow(new codes.ERROR_PAYLOAD_TOO_LARGE({ messageValues: ['underlying DB provider'], sdkDetails: copyParams }))
case 429:
return logAndThrow(new codes.ERROR_REQUEST_RATE_TOO_HIGH({ sdkDetails: copyParams }))
default: // 500 errors
return logAndThrow(new codes.ERROR_INTERNAL({ messageValues: [`unexpected response from provider with status: ${response.status}`], sdkDetails: copyParams }))
shazron marked this conversation as resolved.
Show resolved Hide resolved
}
}

/**
* @abstract
* @class AdobeState
Expand Down Expand Up @@ -263,7 +268,7 @@ class AdobeState {
logger.debug('get', requestOptions)
const promise = this.fetchRetry.exponentialBackoff(this.createRequestUrl(key), requestOptions)
const response = await _wrap(promise, { key })
if (response) {
if (response.ok) {
// we only expect string values
const value = await response.text()
const expiration = new Date(Number(response.headers.get(HEADER_KEY_EXPIRES))).toISOString()
Expand Down Expand Up @@ -327,7 +332,7 @@ class AdobeState {
* Deletes a state key-value pair
*
* @param {string} key state key identifier
* @returns {Promise<string>} key of deleted state or `null` if state does not exists
* @returns {Promise<string>} key of deleted state or `null` if state does not exist
* @memberof AdobeState
*/
async delete (key) {
Expand All @@ -343,8 +348,12 @@ class AdobeState {
logger.debug('delete', requestOptions)

const promise = this.fetchRetry.exponentialBackoff(this.createRequestUrl(key), requestOptions)
const ret = await _wrap(promise, { key })
return ret && key
const response = await _wrap(promise, { key })
if (response.status === 404) {
return null
} else {
return key
}
}

/**
Expand All @@ -365,7 +374,7 @@ class AdobeState {

const promise = this.fetchRetry.exponentialBackoff(this.createRequestUrl(), requestOptions)
const response = await _wrap(promise, {})
return response !== null
return (response.status !== 404)
}

/**
Expand All @@ -386,7 +395,7 @@ class AdobeState {

const promise = this.fetchRetry.exponentialBackoff(this.createRequestUrl(), requestOptions)
const response = await _wrap(promise, {})
return response !== null
return (response.status !== 404)
}

/**
Expand All @@ -407,7 +416,11 @@ class AdobeState {

const promise = this.fetchRetry.exponentialBackoff(this.createRequestUrl(), requestOptions)
const response = await _wrap(promise, {})
return !!response && response.json()
if (response.status === 404) {
return false
} else {
return response.json()
}
}
}

Expand Down
7 changes: 3 additions & 4 deletions test/AdobeState.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -263,9 +263,8 @@ describe('put', () => {
const value = 'some-value'

const error = new Error('some network error')
error.code = 502
mockExponentialBackoff.mockRejectedValue(error)
await expect(store.put(key, value)).rejects.toThrow('[AdobeStateLib:ERROR_INTERNAL] unexpected response from provider with status: 502')
await expect(store.put(key, value)).rejects.toThrow(error.message)
})
})

Expand Down Expand Up @@ -319,7 +318,7 @@ describe('deleteAll', () => {
})
})

describe('any', () => {
describe('stats()', () => {
let store

beforeEach(async () => {
Expand All @@ -342,7 +341,7 @@ describe('any', () => {
})
})

describe('stats()', () => {
describe('any', () => {
let store

beforeEach(async () => {
Expand Down
Loading