Skip to content

Controllers

Hishel provides the Controllers, which allow you to fully customize how the cache works at the specification level.

You can choose which parts of RFC 9111 to ignore. For example, this is useful when you want to ensure that your client does not use stale responses even if they are acceptable from the server.

Force caching

If you only need to cache responses without validating the headers and following RFC9111 rules, simply set the force_cache property to true.

Example:

import hishel

controller = hishel.Controller(force_cache=True)
client = hishel.CacheClient(controller=controller)

Note

force_cache extension will always overwrite the controller's force_cache property.

Cachable HTTP methods

You can specify which HTTP methods Hishel should cache.

Example:

import hishel

controller = hishel.Controller(cacheable_methods=["GET", "POST"])
client = hishel.CacheClient(controller=controller)

Note

Hishel will only cache GET methods if the cachable methods are not explicitly specified.

Cachable status codes

If you only want to cache specific status codes, do so.

Example:

import hishel

controller = hishel.Controller(cacheable_status_codes=[301, 308])
client = hishel.CacheClient(controller=controller)

Note

If the cachable status codes are not explicitly specified, Hishel will only cache status codes 200, 301, and 308.

Allowing heuristics

You can enable heuristics calculations, which are disabled by default.

Example:

import hishel

controller = hishel.Controller(allow_heuristics=True)
client = hishel.CacheClient(controller=controller)

Hishel is very conservative about what status codes are permitted to be heuristically cacheable. When allow_heuristics is enabled, Hishel will only cache responses having status codes 200, 301, and 308. In contrast, RFC 9111 specifies that many more responses can be heuristically cacheable, specifically 200, 203, 204, 206, 300, 301, 308, 404, 405, 410, 414, and 501.

If you would prefer heuristic caching to the fullest extent permitted by RFC 9111, then pass HEURISTICALLY_CACHEABLE_STATUS_CODES to cacheable_status_codes:

import hishel

controller = hishel.Controller(
    allow_heuristics=True,
    cacheable_status_codes=hishel.HEURISTICALLY_CACHEABLE_STATUS_CODES
    )
client = hishel.CacheClient(controller=controller)

Tip

If you're not familiar with Heuristics Caching, you can read about it in the specification.

Preventing caching of private responses

If you want Hishel to act as a shared cache, you need to prevent it from caching responses with the private directive.

Example:

import hishel

controller = hishel.Controller(cache_private=False)
client = hishel.CacheClient(controller=controller)

Note

Servers may prohibit only some headers from being stored in a shared cache by sending a headers such as Cache-Control: private=set-cookie. However, Hishel with cache_private=False will still not cache the response, at all.

Allowing stale responses

Some servers allow the use of stale responses if they cannot be re-validated or the client is disconnected from the server. Clients MAY use stale responses in such cases, but this behavior is disabled by default in Hishel.

Example:

import hishel

controller = hishel.Controller(allow_stale=True)
client = hishel.CacheClient(controller=controller)

Tip

Hishel will attempt to use stale response only if the client is unable to connect to the server to make a request. You can enable stale responses to receive responses even if your internet connection is lost.

Specifying revalidation behavior

Responses are revalidated by default when they become stale; however, you can always revalidate the responses if you wish.

Example:

import hishel

controller = hishel.Controller(always_revalidate=True)
client = hishel.CacheClient(controller=controller)

Note

Because we already have the response body in our cache, revalidation is very quick.

Custom cache keys

By default, Hishel generates cache keys as a hash of the request method and url. However, you can customize cache key creation by writing a function with the signature Callable[[httpcore.Request], str] and passing it to the controller.

Example:

import hishel
import httpcore
from hishel._utils import generate_key

def custom_key_generator(request: httpcore.Request, body: bytes):
    key = generate_key(request, body)
    method = request.method.decode()
    host = request.url.host.decode()
    return f"{method}|{host}|{key}"

controller = hishel.Controller(key_generator=custom_key_generator)
client = hishel.CacheClient(controller=controller)

client.get("https://hishel.com")

Instead of just the hashed_value, the key now has the format method|host|hashed_value.

Note

Cache keys are used to store responses in storages, such as filesystem storage, which will use the cache key to create a file with that value. You can write your own cache key implementation to have more meaningful file names and simplify cache monitoring.

📁 root
└─╴📁 .cache
   └─╴📁 hishel
      └─╴📄 41ebb4dd16761e94e2ee36b71e0d916e
📁 root
└─╴📁 .cache
   └─╴📁 hishel
      └─╴📄 GET|hishel.com|41ebb4dd16761e94e2ee36b71e0d916e