memoize

package
v1.0.6 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: May 24, 2023 License: MIT Imports: 12 Imported by: 0

README

Memoize

Why context?

As a matter of fact, it's not always possible to cache some data across API calls. At times, there's a need to memoize the result of a function call within the boundary of only one single API call. For example, let's say we call a Map service to get the distance to travel from point A to point B. Across different API calls at different points in time, the resulting distance might be different due to changing traffic conditions. Hence, it's only feasible to cache the result within a single API call so that all logic in this boundary uses the same distance value.

In this case, context is a perfect candidate for storing such request-level data.

How to use

At the beginning of your API handling logic, you must initialize a cache using the following function.

// WithCache returns a new context.Context that holds a reference to
// a cache for memoized functions. This is meant to be a request-level
// cache that will automatically get garbage-collected at the end of
// an API request when the context itself is garbage-collected.
//
// WithCache must be called near the start of an API request handling
// before being any memoized functions get executed in child goroutines.
//
// The given context will be used as the root context of this cache. If
// it gets cancelled, all pending memoized executions will be abandoned.
// On the other hand, the context given to Execute won't affect pending
// executions. Child goroutines can cancel the context given to Execute
// to stop waiting for the result from the memoized function, which will
// still proceed till completion.
//
// Note: the return DestroyFn must be deferred to minimize memory leaks.
func WithCache(ctx context.Context) (context.Context, DestroyFn)

After that, depending on your implementation, you can optionally pre-populate the memoize cache using the function below.

// PopulateCache will put the given entries into this cache. The key
// of such entries should be the executionKey that would be used to
// call execute. The value should be the Outcome that you want to map
// to this executionKey.
//
// Note: the given entries can only be populated in the cache if the
// input context has been initialized using WithCache.
func PopulateCache(ctx context.Context, entries map[interface{}]Outcome)

Subsequently, you can pass the context you got back from the above function down to lower-level code. Whenever there's a need to memoize function calls, you just need to execute those functions using the provided function below.

// Execute guarantees that the given memoizedFn will be invoked only
// once regardless of how many times Execute gets called with the same
// executionKey. All callers will receive the same result and error as
// the result of this call.
//
// Note 1: this promise can only be kept if the given context has been
// initialized using WithCache before calling Execute.
//
// Note 2: the provided key must be comparable and should not be of type
// string or any other built-in type to avoid collisions between packages
// using this context. Callers of Execute should define their own types
// for keys similar to the best practices for using context.WithValue.
//
// Note 3: cancelling the given context allows caller to stop waiting
// for the result from the memoizedFn. However, the memoizedFn will
// still proceed till completion unless the root context given to
// WithCache was cancelled.
func Execute[K comparable, V any](
    ctx context.Context,
    executionKey K,
    memoizedFn func(context.Context) (V, error),
) (TypedOutcome[V], Extra)

Toward the end of your implementation, if there's a need to find all memoized outcomes related to a particular execution key type (e.g. to put them in Redis so that they can be used to pre-populate the cache for subsequent requests), you can take advantage of the FindOutcomes function.

// FindOutcomes returns all Outcome that were memoized under the given
// executionKey type at the time findOutcomes was called. If a promise
// related to this executionKey type is still pending, the function
// will block and wait for it to complete to get its Outcome.
//
// Note: this function can only return all memoized Outcome if the given
// context has been initialized using WithCache.
func FindOutcomes[K comparable, V any](ctx context.Context, executionKey K) map[K]TypedOutcome[V]

Documentation

Index

Constants

This section is empty.

Variables

View Source
var (
	ErrPanicExecutingMemoizedFn = errors.New("panic executing memoizedFn")
	ErrCacheAlreadyDestroyed    = errors.New("cache already destroyed, cannot be used anymore")
	ErrMemoizedFnCannotBeNil    = errors.New("memoizedFn cannot be nil")
)

Functions

func Execute

func Execute[K comparable, V any](
	ctx context.Context,
	executionKey K,
	memoizedFn func(context.Context) (V, error),
) (TypedOutcome[V], Extra)

Execute guarantees that the given memoizedFn will be invoked only once regardless of how many times Execute gets called with the same executionKey. All callers will receive the same result and error as the result of this call.

Note 1: this promise can only be kept if the given context has been initialized using WithCache before calling Execute.

Note 2: the provided key must be comparable and should not be of type string or any other built-in type to avoid collisions between packages using this context. Callers of Execute should define their own types for keys similar to the best practices for using context.WithValue.

Note 3: cancelling the given context allows caller to stop waiting for the result from the memoizedFn. However, the memoizedFn will still proceed till completion unless the root context given to WithCache was cancelled.

func FindOutcomes added in v1.0.3

func FindOutcomes[K comparable, V any](ctx context.Context, executionKey K) map[K]TypedOutcome[V]

FindOutcomes returns all Outcome that were memoized under the given executionKey type at the time findOutcomes was called. If a promise related to this executionKey type is still pending, the function will block and wait for it to complete to get its Outcome.

Note: this function can only return all memoized Outcome if the given context has been initialized using WithCache.

func PopulateCache added in v1.0.3

func PopulateCache(ctx context.Context, entries map[interface{}]Outcome)

PopulateCache will put the given entries into this cache. The key of such entries should be the executionKey that would be used to call execute. The value should be the Outcome that you want to map to this executionKey.

Note: the given entries can only be populated in the cache if the input context has been initialized using WithCache.

Types

type DestroyFn

type DestroyFn func()

func WithCache

func WithCache(ctx context.Context) (context.Context, DestroyFn)

WithCache returns a new context.Context that holds a reference to a cache for memoized functions. This is meant to be a request-level cache that will automatically get garbage-collected at the end of an API request when the context itself is garbage-collected.

WithCache must be called near the start of an API request handling before being any memoized functions get executed in child goroutines.

The given context will be used as the root context of this cache. If it gets cancelled, all pending memoized executions will be abandoned. On the other hand, the context given to Execute won't affect pending executions. Child goroutines can cancel the context given to Execute to stop waiting for the result from the memoized function, which will still proceed till completion.

Note: the return DestroyFn must be deferred to minimize memory leaks.

func WithConcurrentCache added in v1.0.6

func WithConcurrentCache(ctx context.Context, concurrencyLevel int) (context.Context, DestroyFn)

WithConcurrentCache returns a new context.Context that holds a reference to a cache for memoized functions. This is meant to be a request-level cache that will automatically get garbage-collected at the end of an API request when the context itself is garbage-collected.

WithConcurrentCache must be called at the start of an API request handling before being any memoized functions get executed in child goroutines.

The given context will be used as the root context of this cache. If it is cancelled, all pending memoized executions will be abandoned. In contrast, the context given to Execute won't affect pending executions. Child routines can cancel the context given to Execute to stop waiting for the result from the memoized function, which will still proceed till completion.

Note: the return DestroyFn must be deferred to minimize memory leaks.

type Extra added in v1.0.3

type Extra struct {
	// IsMemoized indicates if the outcome was memoized.
	IsMemoized bool
	// IsExecuted indicates if the outcome came from actual execution or
	// was pre-populated in the cache.
	IsExecuted bool
}

Extra includes additional details about the returned outcome.

type Function

type Function func(ctx context.Context) (interface{}, error)

Function is the type of function that can be memoized.

The argument must not materially affect the result of the function in ways that are not captured by the promise's key, since if promise.get is called twice concurrently, with the same (implicit) key but different arguments, the Function is called once and only once but its result must be suitable for both callers.

type Outcome added in v1.0.3

type Outcome struct {
	Value interface{}
	Err   error
}

Outcome is the outcome of executing a memoized function.

type TypedOutcome added in v1.0.4

type TypedOutcome[V any] struct {
	Value V
	Err   error
}

TypedOutcome ...

func (TypedOutcome[V]) ResultOrDefault added in v1.0.5

func (o TypedOutcome[V]) ResultOrDefault(defaultResult V) V

ResultOrDefault returns the final result if there's no error or the default result if there's an error.

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL