Caddy Module: http.handlers.cache

September 11, 2024 ยท View on GitHub

This is a distributed HTTP cache module for Caddy based on Souin cache.

Warning

Since v1.7.0 Souin (the development repository that cache-handler is based on) implements only one storage. If you need a specific storage you have to take it from the storages repository and add it either in your code, during the build otherwise.
(e.g. with otter using caddy) You have to build your caddy module with the desired storage xcaddy build --with github.com/caddyserver/cache-handler --with github.com/darkweak/storages/otter/caddy and configure otter in your Caddyfile/JSON configuration file.
See the documentation about the storages.

Features

Minimal Configuration

Using the minimal configuration the responses will be cached for 120s

{
    cache
}

example.com {
    cache
    reverse_proxy your-app:8080
}

Global Option Syntax

Here are all the available options for the global options

{
    log {
        level debug
    }
    cache {
        allowed_http_verbs GET POST PATCH
        api {
            basepath /some-basepath
            prometheus
            souin {
                basepath /souin-changed-endpoint-path
            }
        }
        badger {
            path the_path_to_a_file.json
        }
        cache_keys {
            .*\.css {
                disable_body
                disable_host
                disable_method
                disable_query
                headers X-Token Authorization
                hide
            }
        }
        cache_name Another
        cdn {
            api_key XXXX
            dynamic
            email darkweak@protonmail.com
            hostname domain.com
            network your_network
            provider fastly
            strategy soft
            service_id 123456_id
            zone_id anywhere_zone
        }
        etcd {
            configuration {
                # Your etcd configuration here
            }
        }
        key {
            disable_body
            disable_host
            disable_method
            headers Content-Type Authorization
        }
        log_level debug
        mode bypass
        nuts {
            path /path/to/the/storage
        }
        olric {
            url url_to_your_cluster:3320
            path the_path_to_a_file.yaml
            configuration {
                # Your olric configuration here
            }
        }
        regex {
            exclude /test2.*
        }
        stale 200s
        ttl 1000s
        default_cache_control no-store
    }
}

:4443
respond "Hello World!"

Cache directive Syntax

Here are all the available options for the directive options

@match path /path

handle @match {
    cache {
        cache_name ChangeName
        cache_keys {
            (host1|host2).*\.css {
                disable_body
                disable_host
                disable_method
                disable_query
                headers X-Token Authorization
            }
        }
        cdn {
            api_key XXXX
            dynamic
            email darkweak@protonmail.com
            hostname domain.com
            network your_network
            provider fastly
            strategy soft
            service_id 123456_id
            zone_id anywhere_zone
        }
        key {
            disable_body
            disable_host
            disable_method
            disable_query
            headers Content-Type Authorization
        }
        log_level debug
        regex {
            exclude /test2.*
        }
        stale 200s
        ttl 1000s
        default_cache_control no-store
    }
}

Provider Syntax

Badger

The badger provider must have either the path or the configuration directive.

badger-path.com {
    cache {
        badger {
            path /tmp/badger/first-match
        }
    }
}
badger-configuration.com {
    cache {
        badger {
            configuration {
                # Required value
                ValueDir <string>

                # Optional
                SyncWrites <bool>
                NumVersionsToKeep <int>
                ReadOnly <bool>
                Compression <int>
                InMemory <bool>
                MetricsEnabled <bool>
                MemTableSize <int>
                BaseTableSize <int>
                BaseLevelSize <int>
                LevelSizeMultiplier <int>
                TableSizeMultiplier <int>
                MaxLevels <int>
                VLogPercentile <float>
                ValueThreshold <int>
                NumMemtables <int>
                BlockSize <int>
                BloomFalsePositive <float>
                BlockCacheSize <int>
                IndexCacheSize <int>
                NumLevelZeroTables <int>
                NumLevelZeroTablesStall <int>
                ValueLogFileSize <int>
                ValueLogMaxEntries <int>
                NumCompactors <int>
                CompactL0OnClose <bool>
                LmaxCompaction <bool>
                ZSTDCompressionLevel <int>
                VerifyValueChecksum <bool>
                EncryptionKey <string>
                EncryptionKeyRotationDuration <Duration>
                BypassLockGuard <bool>
                ChecksumVerificationMode <int>
                DetectConflicts <bool>
                NamespaceOffset <int>
            }
        }
    }
}

Etcd

The etcd provider must have the configuration directive.

etcd-configuration.com {
    cache {
        etcd {
            configuration {
                Endpoints etcd1:2379 etcd2:2379 etcd3:2379
                AutoSyncInterval 1s
                DialTimeout 1s
                DialKeepAliveTime 1s
                DialKeepAliveTimeout 1s
                MaxCallSendMsgSize 10000000
                MaxCallRecvMsgSize 10000000
                Username john
                Password doe
                RejectOldCluster false
                PermitWithoutStream false
            }
        }
    }
}

NutsDB

The nutsdb provider must have either the path or the configuration directive.

nuts-path.com {
    cache {
        nuts {
            path /tmp/nuts-path
        }
    }
}
nuts-configuration.com {
    cache {
        nuts {
            configuration {
                Dir /tmp/nuts-configuration
                EntryIdxMode 1
                RWMode 0
                SegmentSize 1024
                NodeNum 42
                SyncEnable true
                StartFileLoadingMode 1
            }
        }
    }
}

Olric

The olric provider must have either the url directive to work as client mode.

olric-url.com {
    cache {
        olric {
            url olric:3320
        }
    }
}

The olric provider must have either the path or the configuration directive to work as embedded mode.

olric-path.com {
    cache {
        olric {
            path /path/to/olricd.yml
        }
    }
}
olric-configuration.com {
    cache {
        nuts {
            configuration {
                Dir /tmp/nuts-configuration
                EntryIdxMode 1
                RWMode 0
                SegmentSize 1024
                NodeNum 42
                SyncEnable true
                StartFileLoadingMode 1
            }
        }
    }
}

Redis

The redis provider must have either the URL or the configuration directive.

redis-url.com {
    cache {
        redis {
            url 127.0.0.1:6379
        }
    }
}

You can also use the configuration. Refer to the Souin docs or rueidis client options to define your config as key value.

What does these directives mean?

KeyDescriptionValue example
allowed_http_verbsThe HTTP verbs allowed to be cachedGET POST PATCH

(default: GET HEAD)
apiThe cache-handler API cache management
api.basepathBasePath for all APIs to avoid conflicts/your-non-conflict-route

(default: /souin-api)
api.prometheusEnable the Prometheus metrics
api.souin.basepathSouin API basepath/another-souin-api-route

(default: /souin)
badgerConfigure the Badger cache storage
badger.pathConfigure Badger with a file/anywhere/badger_configuration.json
badger.configurationConfigure Badger directly in the Caddyfile or your JSON caddy configurationSee the Badger configuration for the options
cache_nameOverride the cache name to use in the Cache-Status response headerAnother Caddy Cache-Handler Souin
cache_keysDefine the key generation rules for each URI matching the key regexp
cache_keys.{your regexp}Regexp that the URI should match to override the key generation.+\.css
cache_keys.{your regexp}Regexp that the URI should match to override the key generation.+\.css
cache_keys.{your regexp}.disable_bodyDisable the body part in the key matching the regexp (GraphQL context)true

(default: false)
cache_keys.{your regexp}.disable_hostDisable the host part in the key matching the regexptrue

(default: false)
cache_keys.{your regexp}.disable_methodDisable the method part in the key matching the regexptrue

(default: false)
cache_keys.{your regexp}.disable_queryDisable the query string part in the key matching the regexptrue

(default: false)
cache_keys.{your regexp}.headersAdd headers to the key matching the regexpAuthorization Content-Type X-Additional-Header
cache_keys.{your regexp}.hidePrevent the key from being exposed in the Cache-Status HTTP response headertrue

(default: false)
cdnThe CDN management, if you use any cdn to proxy your requests Souin will handle that
cdn.providerThe provider placed before Souinakamai

fastly

souin
cdn.api_keyThe api key used to access to the providerXXXX
cdn.dynamicEnable the dynamic keys returned by your backend application(default: true)
cdn.emailThe api key used to access to the provider if required, depending the providerXXXX
cdn.hostnameThe hostname if required, depending the providerdomain.com
cdn.networkThe network if required, depending the provideryour_network
cdn.strategyThe strategy to use to purge the cdn cache, soft will keep the content as a stale resourcehard

(default: soft)
cdn.service_idThe service id if required, depending the provider123456_id
cdn.zone_idThe zone id if required, depending the provideranywhere_zone
default_cache_controlSet the default value of Cache-Control response header if not set by upstream (Souin treats empty Cache-Control as public if omitted)no-store
keyOverride the key generation with the ability to disable unecessary parts
key.disable_bodyDisable the body part in the key (GraphQL context)true

(default: false)
key.disable_hostDisable the host part in the keytrue

(default: false)
key.disable_methodDisable the method part in the keytrue

(default: false)
key.disable_queryDisable the query string part in the keytrue

(default: false)
key.disable_schemeDisable the scheme string part in the keytrue

(default: false)
key.hashHash the key before store it in the storage to get smaller keystrue

(default: false)
key.headersAdd headers to the key matching the regexpAuthorization Content-Type X-Additional-Header
key.hidePrevent the key from being exposed in the Cache-Status HTTP response headertrue

(default: false)
key.templateUse caddy templates to create the key (when this option is enabled, disable_* directives are skipped)KEY-{http.request.uri.path}-{http.request.uri.query}
max_cacheable_body_bytesSet the maximum size (in bytes) for a response body to be cached (unlimited if omited)1048576 (1MB)
modeBypass the RFC respectOne of bypass bypass_request bypass_response strict (default strict)
nutsConfigure the Nuts cache storage
nuts.pathSet the Nuts file path storage/anywhere/nuts/storage
nuts.configurationConfigure Nuts directly in the Caddyfile or your JSON caddy configurationSee the Nuts configuration for the options
etcdConfigure the Etcd cache storage
etcd.configurationConfigure Etcd directly in the Caddyfile or your JSON caddy configurationSee the Etcd configuration for the options
olricConfigure the Olric cache storage
olric.pathConfigure Olric with a file/anywhere/olric_configuration.json
olric.configurationConfigure Olric directly in the Caddyfile or your JSON caddy configurationSee the Olric configuration for the options
otterConfigure the Otter cache storage
otter.configurationConfigure Otter directly in the Caddyfile or your JSON caddy configuration
otter.configuration.sizeSet the size of the pool in Otter999999 (default 10000)
redisConfigure the Redis cache storage
redis.urlSet the Redis url storagelocalhost:6379
redis.configurationConfigure Redis directly in the Caddyfile or your JSON caddy configurationSee the Nuts configuration for the options
regex.excludeThe regex used to prevent paths being cached^[A-z]+.*$
staleThe stale duration25m
storersStorers chain to fallback if a previous one is unreachable or don't have the resourceotter nuts badger redis
timeoutThe timeout configuration
timeout.backendThe timeout duration to consider the backend as unreachable10s
timeout.cacheThe timeout duration to consider the cache provider as unreachable10ms
ttlThe TTL duration120s
log_levelThe log levelOne of DEBUG, INFO, WARN, ERROR, DPANIC, PANIC, FATAL it's case insensitive

Other resources

You can find an example for the Caddyfile or the JSON file.
See the Souin configuration for the full configuration, and its associated Caddyfile

Development and Stable Versions

The Souin repository serves as the development version, where new features are introduced and tested. Once these features have been thoroughly stabilized, they are integrated into the cache-handler repository through a dependency update. This ensures that cache-handler remains the stable and reliable version for production use.