-
-
Notifications
You must be signed in to change notification settings - Fork 1.1k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(core): Remove all long-lived in-memory state, use short-TTL caching
Relates to #988. This change resolves stale data issues when running multiple Vendure instances in parallel, e.g. as serverless functions.
- Loading branch information
1 parent
214b86b
commit d428ffc
Showing
23 changed files
with
157 additions
and
83 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,73 @@ | ||
import { Logger } from '../config/logger/vendure-logger'; | ||
|
||
/** | ||
* @description | ||
* A cache which automatically refreshes itself if the value is found to be stale. | ||
*/ | ||
export interface SelfRefreshingCache<V> { | ||
/** | ||
* @description | ||
* The current value of the cache. If the value is stale, the data will be refreshed and then | ||
* the fresh value will be returned. | ||
*/ | ||
value(): Promise<V>; | ||
|
||
/** | ||
* @description | ||
* Force a refresh of the value, e.g. when it is known that the value has changed such as after | ||
* an update operation to the source data in the database. | ||
*/ | ||
refresh(): Promise<V>; | ||
} | ||
|
||
export interface SelfRefreshingCacheConfig<V> { | ||
name: string; | ||
ttl: number; | ||
refreshFn: () => Promise<V>; | ||
} | ||
|
||
/** | ||
* @description | ||
* Creates a {@link SelfRefreshingCache} object, which is used to cache a single frequently-accessed value. In this type | ||
* of cache, the function used to populate the value (`refreshFn`) is defined during the creation of the cache, and | ||
* it is immediately used to populate the initial value. | ||
* | ||
* From there, when the `.value` property is accessed, it will _always_ return a value synchronously. If the | ||
* value has expired, it will still be returned and the `refreshFn` will be triggered to update the value in the | ||
* background. | ||
*/ | ||
export async function createSelfRefreshingCache<V>( | ||
config: SelfRefreshingCacheConfig<V>, | ||
): Promise<SelfRefreshingCache<V>> { | ||
const { ttl, name, refreshFn } = config; | ||
const initialValue = await refreshFn(); | ||
let value = initialValue; | ||
let expires = new Date().getTime() + ttl; | ||
const refreshValue = (): Promise<V> => { | ||
Logger.debug(`Refreshing the SelfRefreshingCache "${name}"`); | ||
return refreshFn() | ||
.then(newValue => { | ||
value = newValue; | ||
expires = new Date().getTime() + ttl; | ||
return value; | ||
}) | ||
.catch(err => { | ||
Logger.error( | ||
`Failed to update SelfRefreshingCache "${name}": ${err.message}`, | ||
undefined, | ||
err.stack, | ||
); | ||
return value; | ||
}); | ||
}; | ||
return { | ||
async value() { | ||
const now = new Date().getTime(); | ||
if (expires < now) { | ||
return refreshValue(); | ||
} | ||
return value; | ||
}, | ||
refresh: refreshValue, | ||
}; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.