Skip to content

Commit 55ccd06

Browse files
authored
feat: add memory cache backend (#7048)
Signed-off-by: knqyf263 <[email protected]>
1 parent 14d71ba commit 55ccd06

16 files changed

+577
-42
lines changed

docs/docs/configuration/cache.md

+40-10
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ The cache option is common to all scanners.
1111
## Clear Caches
1212
`trivy clean` subcommand removes caches.
1313

14-
```
14+
```bash
1515
$ trivy clean --scan-cache
1616
```
1717

@@ -31,31 +31,59 @@ See `trivy clean --help` for details.
3131
## Cache Directory
3232
Specify where the cache is stored with `--cache-dir`.
3333

34-
```
34+
```bash
3535
$ trivy --cache-dir /tmp/trivy/ image python:3.4-alpine3.9
3636
```
3737

38-
## Cache Backend
38+
## Scan Cache Backend
3939
!!! warning "EXPERIMENTAL"
4040
This feature might change without preserving backwards compatibility.
4141

42-
Trivy supports local filesystem and Redis as the cache backend. This option is useful especially for client/server mode.
43-
44-
Two options:
42+
Trivy utilizes a scan cache to store analysis results, such as package lists.
43+
It supports three types of backends for this cache:
4544

46-
- `fs`
47-
- the cache path can be specified by `--cache-dir`
48-
- `redis://`
45+
- Local File System (`fs`)
46+
- The cache path can be specified by `--cache-dir`
47+
- Memory (`memory`)
48+
- Redis (`redis://`)
4949
- `redis://[HOST]:[PORT]`
5050
- TTL can be configured via `--cache-ttl`
5151

52+
### Local File System
53+
The local file system backend is the default choice for container and VM image scans.
54+
When scanning container images, it stores analysis results on a per-layer basis, using layer IDs as keys.
55+
This approach enables faster scans of the same container image or different images that share layers.
56+
57+
!!! note
58+
Internally, this backend uses [BoltDB][boltdb], which has an important limitation: only one process can access the cache at a time.
59+
Subsequent processes attempting to access the cache will be locked.
60+
For more details on this limitation, refer to the [troubleshooting guide][parallel-run].
61+
62+
### Memory
63+
The memory backend stores analysis results in memory, which means the cache is discarded when the process ends.
64+
This makes it useful in scenarios where caching is not required or desired.
65+
It serves as the default for repository, filesystem and SBOM scans and can also be employed for container image scans when caching is unnecessary.
66+
67+
To use the memory backend for a container image scan, you can use the following command:
68+
69+
```bash
70+
$ trivy image debian:11 --cache-backend memory
5271
```
72+
73+
### Redis
74+
75+
The Redis backend is particularly useful when you need to share the cache across multiple Trivy instances.
76+
You can set up Trivy to use a Redis backend with a command like this:
77+
78+
```bash
5379
$ trivy server --cache-backend redis://localhost:6379
5480
```
5581

82+
This approach allows for centralized caching, which can be beneficial in distributed or high-concurrency environments.
83+
5684
If you want to use TLS with Redis, you can enable it by specifying the `--redis-tls` flag.
5785

58-
```shell
86+
```bash
5987
$ trivy server --cache-backend redis://localhost:6379 --redis-tls
6088
```
6189

@@ -72,6 +100,8 @@ $ trivy server --cache-backend redis://localhost:6379 \
72100
[trivy-db]: ./db.md#vulnerability-database
73101
[trivy-java-db]: ./db.md#java-index-database
74102
[misconf-checks]: ../scanner/misconfiguration/check/builtin.md
103+
[boltdb]: https://github.com/etcd-io/bbolt
104+
[parallel-run]: https://aquasecurity.github.io/trivy/v0.52/docs/references/troubleshooting/#running-in-parallel-takes-same-time-as-series-run
75105

76106
[^1]: Downloaded when scanning for vulnerabilities
77107
[^2]: Downloaded when scanning `jar/war/par/ear` files

docs/docs/references/configuration/cli/trivy_config.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ trivy config [flags] DIR
99
### Options
1010

1111
```
12-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
12+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "memory")
1313
--cache-ttl duration cache TTL when using redis as cache backend
1414
--cf-params strings specify paths to override the CloudFormation parameters files
1515
--check-namespaces strings Rego namespaces

docs/docs/references/configuration/cli/trivy_filesystem.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ trivy filesystem [flags] PATH
1919
### Options
2020

2121
```
22-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
22+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "memory")
2323
--cache-ttl duration cache TTL when using redis as cache backend
2424
--cf-params strings specify paths to override the CloudFormation parameters files
2525
--check-namespaces strings Rego namespaces

docs/docs/references/configuration/cli/trivy_image.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ trivy image [flags] IMAGE_NAME
3434
### Options
3535

3636
```
37-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
37+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "fs")
3838
--cache-ttl duration cache TTL when using redis as cache backend
3939
--check-namespaces strings Rego namespaces
4040
--checks-bundle-repository string OCI registry URL to retrieve checks bundle from (default "ghcr.io/aquasecurity/trivy-checks:0")

docs/docs/references/configuration/cli/trivy_kubernetes.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ trivy kubernetes [flags] [CONTEXT]
3030

3131
```
3232
--burst int specify the maximum burst for throttle (default 10)
33-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
33+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "fs")
3434
--cache-ttl duration cache TTL when using redis as cache backend
3535
--check-namespaces strings Rego namespaces
3636
--checks-bundle-repository string OCI registry URL to retrieve checks bundle from (default "ghcr.io/aquasecurity/trivy-checks:0")

docs/docs/references/configuration/cli/trivy_repository.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ trivy repository [flags] (REPO_PATH | REPO_URL)
1919

2020
```
2121
--branch string pass the branch name to be scanned
22-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
22+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "memory")
2323
--cache-ttl duration cache TTL when using redis as cache backend
2424
--cf-params strings specify paths to override the CloudFormation parameters files
2525
--check-namespaces strings Rego namespaces

docs/docs/references/configuration/cli/trivy_rootfs.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ trivy rootfs [flags] ROOTDIR
2222
### Options
2323

2424
```
25-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
25+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "memory")
2626
--cache-ttl duration cache TTL when using redis as cache backend
2727
--cf-params strings specify paths to override the CloudFormation parameters files
2828
--check-namespaces strings Rego namespaces

docs/docs/references/configuration/cli/trivy_sbom.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ trivy sbom [flags] SBOM_PATH
2020
### Options
2121

2222
```
23-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
23+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "memory")
2424
--cache-ttl duration cache TTL when using redis as cache backend
2525
--compliance string compliance report to generate
2626
--custom-headers strings custom headers in client mode

docs/docs/references/configuration/cli/trivy_server.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ trivy server [flags]
2020
### Options
2121

2222
```
23-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
23+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "fs")
2424
--cache-ttl duration cache TTL when using redis as cache backend
2525
--db-repository string OCI repository to retrieve trivy-db from (default "ghcr.io/aquasecurity/trivy-db:2")
2626
--download-db-only download/update vulnerability database but don't run a scan

docs/docs/references/configuration/cli/trivy_vm.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@ trivy vm [flags] VM_IMAGE
2121

2222
```
2323
--aws-region string AWS region to scan
24-
--cache-backend string cache backend (e.g. redis://localhost:6379) (default "fs")
24+
--cache-backend string [EXPERIMENTAL] cache backend (e.g. redis://localhost:6379) (default "fs")
2525
--cache-ttl duration cache TTL when using redis as cache backend
2626
--checks-bundle-repository string OCI registry URL to retrieve checks bundle from (default "ghcr.io/aquasecurity/trivy-checks:0")
2727
--compliance string compliance report to generate

pkg/cache/client.go

+8
Original file line numberDiff line numberDiff line change
@@ -5,12 +5,15 @@ import (
55
"time"
66

77
"golang.org/x/xerrors"
8+
9+
"github.com/aquasecurity/trivy/pkg/log"
810
)
911

1012
const (
1113
TypeUnknown Type = "unknown"
1214
TypeFS Type = "fs"
1315
TypeRedis Type = "redis"
16+
TypeMemory Type = "memory"
1417
)
1518

1619
type Type string
@@ -33,6 +36,8 @@ func NewType(backend string) Type {
3336
return TypeRedis
3437
case backend == "fs", backend == "":
3538
return TypeFS
39+
case backend == "memory":
40+
return TypeMemory
3641
default:
3742
return TypeUnknown
3843
}
@@ -44,6 +49,7 @@ func New(opts Options) (Cache, func(), error) {
4449

4550
var cache Cache
4651
t := NewType(opts.Backend)
52+
log.Debug("Initializing scan cache...", log.String("type", string(t)))
4753
switch t {
4854
case TypeRedis:
4955
redisCache, err := NewRedisCache(opts.Backend, opts.RedisCACert, opts.RedisCert, opts.RedisKey, opts.RedisTLS, opts.TTL)
@@ -58,6 +64,8 @@ func New(opts Options) (Cache, func(), error) {
5864
return nil, cleanup, xerrors.Errorf("unable to initialize fs cache: %w", err)
5965
}
6066
cache = fsCache
67+
case TypeMemory:
68+
cache = NewMemoryCache()
6169
default:
6270
return nil, cleanup, xerrors.Errorf("unknown cache type: %s", t)
6371
}

pkg/cache/memory.go

+98
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,98 @@
1+
package cache
2+
3+
import (
4+
"sync"
5+
6+
"golang.org/x/xerrors"
7+
8+
"github.com/aquasecurity/trivy/pkg/fanal/types"
9+
)
10+
11+
var _ Cache = &MemoryCache{}
12+
13+
type MemoryCache struct {
14+
artifacts sync.Map // Map to store artifact information
15+
blobs sync.Map // Map to store blob information
16+
}
17+
18+
func NewMemoryCache() *MemoryCache {
19+
return &MemoryCache{}
20+
}
21+
22+
// PutArtifact stores the artifact information in the memory cache
23+
func (c *MemoryCache) PutArtifact(artifactID string, artifactInfo types.ArtifactInfo) error {
24+
c.artifacts.Store(artifactID, artifactInfo)
25+
return nil
26+
}
27+
28+
// PutBlob stores the blob information in the memory cache
29+
func (c *MemoryCache) PutBlob(blobID string, blobInfo types.BlobInfo) error {
30+
c.blobs.Store(blobID, blobInfo)
31+
return nil
32+
}
33+
34+
// DeleteBlobs removes the specified blobs from the memory cache
35+
func (c *MemoryCache) DeleteBlobs(blobIDs []string) error {
36+
for _, blobID := range blobIDs {
37+
c.blobs.Delete(blobID)
38+
}
39+
return nil
40+
}
41+
42+
// GetArtifact retrieves the artifact information from the memory cache
43+
func (c *MemoryCache) GetArtifact(artifactID string) (types.ArtifactInfo, error) {
44+
info, ok := c.artifacts.Load(artifactID)
45+
if !ok {
46+
return types.ArtifactInfo{}, xerrors.Errorf("artifact (%s) not found in memory cache", artifactID)
47+
}
48+
artifactInfo, ok := info.(types.ArtifactInfo)
49+
if !ok {
50+
return types.ArtifactInfo{}, xerrors.Errorf("invalid type for artifact (%s) in memory cache", artifactID)
51+
}
52+
return artifactInfo, nil
53+
}
54+
55+
// GetBlob retrieves the blob information from the memory cache
56+
func (c *MemoryCache) GetBlob(blobID string) (types.BlobInfo, error) {
57+
info, ok := c.blobs.Load(blobID)
58+
if !ok {
59+
return types.BlobInfo{}, xerrors.Errorf("blob (%s) not found in memory cache", blobID)
60+
}
61+
blobInfo, ok := info.(types.BlobInfo)
62+
if !ok {
63+
return types.BlobInfo{}, xerrors.Errorf("invalid type for blob (%s) in memory cache", blobID)
64+
}
65+
return blobInfo, nil
66+
}
67+
68+
// MissingBlobs determines the missing artifact and blob information in the memory cache
69+
func (c *MemoryCache) MissingBlobs(artifactID string, blobIDs []string) (bool, []string, error) {
70+
var missingArtifact bool
71+
var missingBlobIDs []string
72+
73+
if _, err := c.GetArtifact(artifactID); err != nil {
74+
missingArtifact = true
75+
}
76+
77+
for _, blobID := range blobIDs {
78+
if _, err := c.GetBlob(blobID); err != nil {
79+
missingBlobIDs = append(missingBlobIDs, blobID)
80+
}
81+
}
82+
83+
return missingArtifact, missingBlobIDs, nil
84+
}
85+
86+
// Close clears the artifact and blob information from the memory cache
87+
func (c *MemoryCache) Close() error {
88+
c.artifacts = sync.Map{}
89+
c.blobs = sync.Map{}
90+
return nil
91+
}
92+
93+
// Clear clears the artifact and blob information from the memory cache
94+
func (c *MemoryCache) Clear() error {
95+
c.artifacts = sync.Map{}
96+
c.blobs = sync.Map{}
97+
return nil
98+
}

0 commit comments

Comments
 (0)