defmodule Cache.Sandbox do
@moduledoc """
Sandbox adapter for isolated testing of applications using ElixirCache.
This module provides a mock implementation of all cache adapters, allowing tests to run in isolation
without interfering with each other's data. The sandbox uses a basic `Agent` to store data in memory
and implements the full range of caching operations supported by all adapters.
## Features
* Isolated cache namespaces for concurrent testing
* Support for all standard cache operations
* Implementation of Redis-specific features like hash and JSON operations
* Implementation of ETS-specific operations for complete testing compatibility
* Lightweight in-memory storage for fast test execution
## Usage
The Sandbox adapter is typically enabled through the `sandbox?` option when defining a cache module:
```elixir
defmodule MyApp.TestCache do
use Cache,
adapter: Cache.Redis, # Original adapter doesn't matter when sandbox is enabled
name: :test_cache,
opts: [],
sandbox?: Mix.env() == :test
end
```
In your tests, use `Cache.SandboxRegistry.start(MyApp.TestCache)` in the setup block to ensure
proper isolation between test cases.
> **Note**: This adapter should not be used in production environments.
"""
use Agent
alias Cache.Redis.JSON
alias Cache.TermEncoder
@behaviour Cache
@impl Cache
def opts_definition, do: []
@impl Cache
def start_link(opts \\ []) do
with {:error, {:already_started, pid}} <- Agent.start_link(fn -> %{} end, opts) do
{:ok, pid}
end
end
@impl Cache
def child_spec({cache_name, opts}) do
%{
id: "#{cache_name}_elixir_cache_agent",
start: {__MODULE__, :start_link, [Keyword.put(opts, :name, cache_name)]}
}
end
@impl Cache
def get(cache_name, key, _opts \\ []) do
Agent.get(cache_name, fn state ->
{:ok, Map.get(state, key)}
end)
end
@impl Cache
def put(cache_name, key, _ttl \\ nil, value, _opts \\ []) do
Agent.update(cache_name, fn state ->
Map.put(state, key, value)
end)
end
@impl Cache
def delete(cache_name, key, _opts \\ []) do
Agent.update(cache_name, fn state ->
Map.delete(state, key)
end)
end
def hash_delete(cache_name, key, hash_key, _opts) do
Agent.update(cache_name, fn state ->
Map.update(state, key, %{}, &Map.delete(&1, hash_key))
end)
end
def hash_get(cache_name, key, hash_key, _opts) do
Agent.get(cache_name, fn state ->
{:ok, state[key][hash_key]}
end)
end
def hash_get_all(cache_name, key, _opts) do
Agent.get(cache_name, fn state ->
case state[key] do
nil -> {:ok, %{}}
value -> {:ok, value}
end
end)
end
def hash_get_many(cache_name, keys_fields, _opts) do
Agent.get(cache_name, fn state ->
values =
Enum.reduce(keys_fields, [], fn {key, fields}, acc ->
values = Enum.map(fields, &state[key][&1])
acc ++ [values]
end)
{:ok, values}
end)
end
def hash_values(cache_name, key, _opts) do
Agent.get(cache_name, fn state ->
{:ok, Map.values(state[key] || %{})}
end)
end
def hash_set(cache_name, key, field, value, ttl, _opts) do
Agent.update(cache_name, fn state ->
put_hash_field_values(state, key, [{field, value}])
end)
if ttl do
{:ok, [1, 1]}
else
:ok
end
end
def hash_set_many(cache_name, keys_fields_values, ttl, _opts) do
Agent.update(cache_name, fn state ->
Enum.reduce(keys_fields_values, state, fn {key, fields_values}, acc ->
put_hash_field_values(acc, key, fields_values)
end)
end)
if ttl do
command_resps =
Enum.map(keys_fields_values, fn {_, fields_values} -> enum_length(fields_values) end)
expiry_resps = Enum.map(keys_fields_values, fn _ -> 1 end)
{:ok, command_resps ++ expiry_resps}
else
:ok
end
end
def json_get(cache_name, key, path, _opts) when path in [nil, ["."]] do
get(cache_name, key)
end
def json_get(cache_name, key, path, _opts) do
if contains_index?(path) do
[index | path] = Enum.reverse(path)
with {:ok, value} <- serialize_path_and_get_value(cache_name, key, path) do
{:ok, Enum.at(value, index)}
end
else
serialize_path_and_get_value(cache_name, key, path)
end
end
def json_set(cache_name, key, path, value, _opts) when path in [nil, ["."]] do
put(cache_name, key, stringify_value(value))
end
def json_set(cache_name, key, path, value, _opts) do
state = Agent.get(cache_name, & &1)
path = JSON.serialize_path(path)
with :ok <- check_key_exists(state, key),
:ok <- check_path_exists(state, key, path) do
path = add_defaults([key | String.split(path, ".")])
value = stringify_value(value)
Agent.update(cache_name, fn state ->
put_in(state, path, value)
end)
end
end
def json_incr(cache_name, key, path, incr \\ 1, _opts) do
Agent.update(cache_name, fn state ->
Map.update(state, key, nil, fn value ->
update_in(value, String.split(path), &(&1 + incr))
end)
end)
end
def json_clear(cache_name, key, path, _opts) do
Agent.update(cache_name, fn state ->
Map.update(
state,
key,
nil,
&update_in(&1, String.split(path, "."), fn
integer when is_integer(integer) -> 0
list when is_list(list) -> []
map when is_map(map) -> %{}
_ -> nil
end)
)
end)
end
def json_delete(cache_name, key, path, _opts) do
Agent.update(cache_name, fn state ->
Map.update(state, key, nil, fn value ->
{_, state} = pop_in(value, String.split(path, "."))
state
end)
end)
end
def json_array_append(cache_name, key, path, value, _opts) do
Agent.update(cache_name, fn state ->
Map.update(state, key, nil, fn state_value ->
update_in(state_value, String.split(path, "."), &(&1 ++ [value]))
end)
end)
end
# Redis Compatibility
def pipeline(_cache_name, _commands, _opts) do
raise "Not Implemented"
end
def pipeline!(_cache_name, _commands, _opts) do
raise "Not Implemented"
end
def command(_cache_name, _command, _opts) do
raise "Not Implemented"
end
def command!(_cache_name, _command, _opts) do
raise "Not Implemented"
end
def scan(cache_name, _scan_opts, _opts) do
Agent.get(cache_name, fn state ->
{:ok, Map.keys(state)}
end)
end
def hash_scan(cache_name, key, _scan_opts, _opts) do
Agent.get(cache_name, fn state ->
case Map.get(state, key) do
nil -> {:ok, []}
map when is_map(map) -> {:ok, Map.keys(map)}
_ -> {:ok, []}
end
end)
end
# ETS & DETS Compatibility
def member(cache_name, key) do
Agent.get(cache_name, fn state ->
Map.has_key?(state, key)
end)
end
def update_counter(cache_name, key, {_pos, incr}) do
Agent.get_and_update(cache_name, fn state ->
current_value = state[key] || 0
new_value = current_value + incr
{new_value, Map.put(state, key, new_value)}
end)
end
def insert_raw(cache_name, data) when is_tuple(data) do
{key, value} = data
Agent.update(cache_name, fn state ->
Map.put(state, key, value)
end)
true
end
def insert_raw(cache_name, data) when is_list(data) do
Agent.update(cache_name, fn state ->
Enum.reduce(data, state, fn {key, value}, acc ->
Map.put(acc, key, value)
end)
end)
true
end
def match_object(_cache_name, _pattern) do
raise "Not Implemented"
end
def match_object(_cache_name, _pattern, _limit) do
raise "Not Implemented"
end
def select(_cache_name, _match_spec) do
raise "Not Implemented"
end
def select(_cache_name, _match_spec, _limit) do
raise "Not Implemented"
end
def info(_cache_name) do
raise "Not Implemented"
end
def info(_cache_name, _item) do
raise "Not Implemented"
end
def select_delete(_cache_name, _match_spec) do
raise "Not Implemented"
end
def match_delete(_cache_name, _pattern) do
raise "Not Implemented"
end
def to_dets(_cache_name, _dets_table) do
raise "Not Implemented"
end
def from_dets(_cache_name, _dets_table) do
raise "Not Implemented"
end
def to_ets(_cache_name) do
raise "Not Implemented"
end
def from_ets(_cache_name, _ets_table) do
raise "Not Implemented"
end
def smembers(_cache_name, _key, _opts) do
raise "Not Implemented"
end
def sadd(_cache_name, _key, _value, _opts) do
raise "Not Implemented"
end
defp put_hash_field_values(state, key, fields_values) do
Map.update(
state,
key,
Map.new(fields_values),
&Enum.reduce(fields_values, &1, fn {field, value}, acc -> Map.put(acc, field, value) end)
)
end
defp check_key_exists(state, key) do
if Map.has_key?(state, key) do
:ok
else
{:error, ErrorMessage.bad_request("ERR new objects must be created at the root")}
end
end
defp check_path_exists(state, key, path) do
case get_in(state, [key | String.split(path, ".")]) do
nil -> {:ok, nil}
_ -> :ok
end
end
defp add_defaults([key | keys]) do
[Access.key(key, key_default(key)) | add_defaults(keys)]
end
defp add_defaults(keys), do: keys
defp key_default(key) do
if Regex.match?(~r/\d+/, key), do: [], else: %{}
end
defp stringify_value(value) do
value
|> TermEncoder.encode_json()
|> TermEncoder.decode_json()
end
defp contains_index?(path) do
path
|> List.last()
|> is_integer()
end
defp serialize_path_and_get_value(cache_name, key, path) do
path = JSON.serialize_path(path)
Agent.get(cache_name, fn state ->
case get_in(state, [key | String.split(path, ".")]) do
nil -> {:error, ErrorMessage.not_found("ERR Path '$.#{path}' does not exist")}
value -> {:ok, value}
end
end)
end
defp enum_length(m) when is_map(m), do: m |> Map.to_list() |> enum_length()
defp enum_length(l), do: length(l)
end