@bluedynamics/cdk8s-plone 0.1.30 → 0.1.32

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/API.md CHANGED
@@ -1794,6 +1794,11 @@ const ploneVinylCacheOptions: PloneVinylCacheOptions = { ... }
1794
1794
  | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.replicas">replicas</a></code> | <code>number</code> | Number of Varnish pod replicas. |
1795
1795
  | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.requestCpu">requestCpu</a></code> | <code>string</code> | CPU request for Varnish pods. |
1796
1796
  | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.requestMemory">requestMemory</a></code> | <code>string</code> | Memory request for Varnish pods. |
1797
+ | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardBy">shardBy</a></code> | <code>string</code> | Shard director: what value is hashed for shard selection. |
1798
+ | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardHealthy">shardHealthy</a></code> | <code>string</code> | Shard director: which backends the director considers when selecting a shard. |
1799
+ | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardRampup">shardRampup</a></code> | <code>string</code> | Shard director: time after adding a backend before it receives its full share of traffic, preventing thundering-herd. |
1800
+ | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardReplicas">shardReplicas</a></code> | <code>number</code> | Shard director: number of Ketama replicas per backend in the hash ring. |
1801
+ | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.storage">storage</a></code> | <code><a href="#@bluedynamics/cdk8s-plone.VinylCacheStorage">VinylCacheStorage</a>[]</code> | Varnish storage backends (`spec.storage`). |
1797
1802
  | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.tolerations">tolerations</a></code> | <code><a href="#@bluedynamics/cdk8s-plone.VinylCacheToleration">VinylCacheToleration</a>[]</code> | Tolerations for the Varnish pods. |
1798
1803
  | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.vclBackendErrorSnippet">vclBackendErrorSnippet</a></code> | <code>string</code> | Custom VCL snippet for vcl_backend_error subroutine. |
1799
1804
  | <code><a href="#@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.vclBackendFetchSnippet">vclBackendFetchSnippet</a></code> | <code>string</code> | Custom VCL snippet for vcl_backend_fetch subroutine. |
@@ -1973,6 +1978,96 @@ Memory request for Varnish pods.
1973
1978
 
1974
1979
  ---
1975
1980
 
1981
+ ##### `shardBy`<sup>Optional</sup> <a name="shardBy" id="@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardBy"></a>
1982
+
1983
+ ```typescript
1984
+ public readonly shardBy: string;
1985
+ ```
1986
+
1987
+ - *Type:* string
1988
+ - *Default:* operator default ("HASH")
1989
+
1990
+ Shard director: what value is hashed for shard selection.
1991
+
1992
+ "HASH" uses the Varnish hash (default); "URL" uses the request URL.
1993
+ Only applied when director is "shard".
1994
+ Requires cloud-vinyl operator >= 0.4.2.
1995
+
1996
+ ---
1997
+
1998
+ ##### `shardHealthy`<sup>Optional</sup> <a name="shardHealthy" id="@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardHealthy"></a>
1999
+
2000
+ ```typescript
2001
+ public readonly shardHealthy: string;
2002
+ ```
2003
+
2004
+ - *Type:* string
2005
+ - *Default:* operator default ("CHOSEN")
2006
+
2007
+ Shard director: which backends the director considers when selecting a shard.
2008
+
2009
+ "CHOSEN" (default) only considers the chosen backend healthy; "ALL" requires all
2010
+ backends to be healthy.
2011
+ Only applied when director is "shard".
2012
+ Requires cloud-vinyl operator >= 0.4.2.
2013
+
2014
+ ---
2015
+
2016
+ ##### `shardRampup`<sup>Optional</sup> <a name="shardRampup" id="@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardRampup"></a>
2017
+
2018
+ ```typescript
2019
+ public readonly shardRampup: string;
2020
+ ```
2021
+
2022
+ - *Type:* string
2023
+ - *Default:* operator default ("30s")
2024
+
2025
+ Shard director: time after adding a backend before it receives its full share of traffic, preventing thundering-herd.
2026
+
2027
+ Only applied when director is "shard".
2028
+
2029
+ ---
2030
+
2031
+ ##### `shardReplicas`<sup>Optional</sup> <a name="shardReplicas" id="@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.shardReplicas"></a>
2032
+
2033
+ ```typescript
2034
+ public readonly shardReplicas: number;
2035
+ ```
2036
+
2037
+ - *Type:* number
2038
+ - *Default:* operator default (67)
2039
+
2040
+ Shard director: number of Ketama replicas per backend in the hash ring.
2041
+
2042
+ Only applied when director is "shard".
2043
+
2044
+ ---
2045
+
2046
+ ##### `storage`<sup>Optional</sup> <a name="storage" id="@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.storage"></a>
2047
+
2048
+ ```typescript
2049
+ public readonly storage: VinylCacheStorage[];
2050
+ ```
2051
+
2052
+ - *Type:* <a href="#@bluedynamics/cdk8s-plone.VinylCacheStorage">VinylCacheStorage</a>[]
2053
+ - *Default:* no storage configured; operator uses varnishd default (~100MB malloc)
2054
+
2055
+ Varnish storage backends (`spec.storage`).
2056
+
2057
+ Each entry becomes a `-s <name>=<type>,<options>` argument to varnishd.
2058
+ If omitted, the operator ships varnishd with its stock default (~100 MB
2059
+ malloc) — almost always too small. Set an explicit malloc size at least
2060
+ matching the pod's memory request to use the allocated memory for caching.
2061
+
2062
+ ---
2063
+
2064
+ *Example*
2065
+
2066
+ ```typescript
2067
+ storage: [{ name: 's0', type: 'malloc', size: '1Gi' }]
2068
+ ```
2069
+
2070
+
1976
2071
  ##### `tolerations`<sup>Optional</sup> <a name="tolerations" id="@bluedynamics/cdk8s-plone.PloneVinylCacheOptions.property.tolerations"></a>
1977
2072
 
1978
2073
  ```typescript
@@ -2365,6 +2460,92 @@ Number of most recent probes to consider.
2365
2460
 
2366
2461
  ---
2367
2462
 
2463
+ ### VinylCacheStorage <a name="VinylCacheStorage" id="@bluedynamics/cdk8s-plone.VinylCacheStorage"></a>
2464
+
2465
+ A Varnish storage backend configuration.
2466
+
2467
+ Maps to `spec.storage[]` on the VinylCache CRD. The operator emits one
2468
+ `-s <name>=<type>,<options>` argument per entry to varnishd.
2469
+
2470
+ Without any storage entry the operator falls back to the varnishd default
2471
+ (~100 MB malloc), which is almost always too small for real workloads.
2472
+
2473
+ #### Initializer <a name="Initializer" id="@bluedynamics/cdk8s-plone.VinylCacheStorage.Initializer"></a>
2474
+
2475
+ ```typescript
2476
+ import { VinylCacheStorage } from '@bluedynamics/cdk8s-plone'
2477
+
2478
+ const vinylCacheStorage: VinylCacheStorage = { ... }
2479
+ ```
2480
+
2481
+ #### Properties <a name="Properties" id="Properties"></a>
2482
+
2483
+ | **Name** | **Type** | **Description** |
2484
+ | --- | --- | --- |
2485
+ | <code><a href="#@bluedynamics/cdk8s-plone.VinylCacheStorage.property.name">name</a></code> | <code>string</code> | Internal storage identifier used in the varnishd `-s` argument. |
2486
+ | <code><a href="#@bluedynamics/cdk8s-plone.VinylCacheStorage.property.type">type</a></code> | <code>string</code> | Storage backend type. |
2487
+ | <code><a href="#@bluedynamics/cdk8s-plone.VinylCacheStorage.property.path">path</a></code> | <code>string</code> | Filesystem path for file-type storage. |
2488
+ | <code><a href="#@bluedynamics/cdk8s-plone.VinylCacheStorage.property.size">size</a></code> | <code>string</code> | Storage allocation as a Kubernetes resource quantity (e.g. "1Gi", "500M"). Required for malloc; required for file. |
2489
+
2490
+ ---
2491
+
2492
+ ##### `name`<sup>Required</sup> <a name="name" id="@bluedynamics/cdk8s-plone.VinylCacheStorage.property.name"></a>
2493
+
2494
+ ```typescript
2495
+ public readonly name: string;
2496
+ ```
2497
+
2498
+ - *Type:* string
2499
+
2500
+ Internal storage identifier used in the varnishd `-s` argument.
2501
+
2502
+ Must be unique within the VinylCache and match `^[a-zA-Z][a-zA-Z0-9_]*$`.
2503
+
2504
+ ---
2505
+
2506
+ ##### `type`<sup>Required</sup> <a name="type" id="@bluedynamics/cdk8s-plone.VinylCacheStorage.property.type"></a>
2507
+
2508
+ ```typescript
2509
+ public readonly type: string;
2510
+ ```
2511
+
2512
+ - *Type:* string
2513
+
2514
+ Storage backend type.
2515
+
2516
+ Only "malloc" and "file" are permitted by the
2517
+ admission webhook.
2518
+
2519
+ ---
2520
+
2521
+ ##### `path`<sup>Optional</sup> <a name="path" id="@bluedynamics/cdk8s-plone.VinylCacheStorage.property.path"></a>
2522
+
2523
+ ```typescript
2524
+ public readonly path: string;
2525
+ ```
2526
+
2527
+ - *Type:* string
2528
+ - *Default:* required for type "file"
2529
+
2530
+ Filesystem path for file-type storage.
2531
+
2532
+ Ignored for malloc.
2533
+
2534
+ ---
2535
+
2536
+ ##### `size`<sup>Optional</sup> <a name="size" id="@bluedynamics/cdk8s-plone.VinylCacheStorage.property.size"></a>
2537
+
2538
+ ```typescript
2539
+ public readonly size: string;
2540
+ ```
2541
+
2542
+ - *Type:* string
2543
+ - *Default:* required for both malloc and file
2544
+
2545
+ Storage allocation as a Kubernetes resource quantity (e.g. "1Gi", "500M"). Required for malloc; required for file.
2546
+
2547
+ ---
2548
+
2368
2549
  ### VinylCacheToleration <a name="VinylCacheToleration" id="@bluedynamics/cdk8s-plone.VinylCacheToleration"></a>
2369
2550
 
2370
2551
  A Kubernetes toleration for the Varnish pods.
@@ -66,6 +66,33 @@ kubectl get pods -n <namespace> -l app.kubernetes.io/managed-by=cloud-vinyl
66
66
 
67
67
  ## Customization
68
68
 
69
+ ### Sizing the Cache Storage
70
+
71
+ Without an explicit `storage` entry, the operator ships varnishd with its
72
+ stock default (~100 MB malloc) — almost always too small. Set a malloc size
73
+ below the pod's memory limit, leaving headroom for varnishd overhead:
74
+
75
+ ```typescript
76
+ new PloneVinylCache(chart, 'cache', {
77
+ plone: plone,
78
+ requestMemory: '512Mi',
79
+ limitMemory: '2Gi',
80
+ storage: [
81
+ { name: 's0', type: 'malloc', size: '1500M' },
82
+ ],
83
+ });
84
+ ```
85
+
86
+ For larger working sets you can combine an in-memory tier with a file-backed
87
+ tier (requires a writable volume mount at the given path):
88
+
89
+ ```typescript
90
+ storage: [
91
+ { name: 'mem', type: 'malloc', size: '500M' },
92
+ { name: 'disk', type: 'file', path: '/var/lib/varnish/disk.bin', size: '10Gi' },
93
+ ]
94
+ ```
95
+
69
96
  ### Custom VCL
70
97
 
71
98
  Override the default Plone VCL snippets for custom caching logic:
@@ -82,6 +109,11 @@ new PloneVinylCache(chart, 'cache', {
82
109
 
83
110
  Invalidation is enabled by default (PURGE, BAN, xkey). Configure `plone.cachepurging` to point to the VinylCache invalidation proxy endpoint.
84
111
 
112
+ :::{note}
113
+ BAN-based invalidation requires **cloud-vinyl operator ≥ 0.4.2**. Earlier versions accept the spec but do not emit the BAN ACL / handler.
114
+ Starting with 0.4.2 the operator's own pod IP is automatically added to the PURGE ACL, so purges from the operator itself work without additional configuration.
115
+ :::
116
+
85
117
  To disable invalidation:
86
118
 
87
119
  ```typescript
@@ -91,6 +123,23 @@ new PloneVinylCache(chart, 'cache', {
91
123
  });
92
124
  ```
93
125
 
126
+ ### Shard Director Tuning
127
+
128
+ For shard-based load distribution (the default), you can fine-tune the consistent-hash behavior. These options require **cloud-vinyl ≥ 0.4.2** to be honored by the generated VCL.
129
+
130
+ ```typescript
131
+ new PloneVinylCache(chart, 'cache', {
132
+ plone: plone,
133
+ director: 'shard',
134
+ shardBy: 'URL', // hash the request URL instead of Varnish's hash
135
+ shardHealthy: 'ALL', // require all backends healthy (vs. only "CHOSEN")
136
+ shardRampup: '45s', // warm-up window for newly added backends
137
+ shardReplicas: 128, // Ketama replicas per backend
138
+ });
139
+ ```
140
+
141
+ Shard options are ignored for non-shard directors (`round_robin`, `random`, `hash`).
142
+
94
143
  ## Migrating from PloneHttpcache
95
144
 
96
145
  Replace `PloneHttpcache` with `PloneVinylCache`:
@@ -307,6 +307,7 @@ Requires the [cloud-vinyl operator](https://github.com/bluedynamics/cloud-vinyl)
307
307
  | `limitCpu` | `string` | No | `500m` | CPU limit |
308
308
  | `requestMemory` | `string` | No | `256Mi` | Memory request |
309
309
  | `limitMemory` | `string` | No | `512Mi` | Memory limit |
310
+ | `storage` | `VinylCacheStorage[]` | No | - | Varnish storage backends. If omitted, the operator falls back to the varnishd default (~100 MB malloc), which is almost always too small. |
310
311
  | `director` | `string` | No | `shard` | Director type: shard, round_robin, random, hash |
311
312
  | `vclRecvSnippet` | `string` | No | built-in | Custom VCL snippet for vcl_recv |
312
313
  | `vclBackendResponseSnippet` | `string` | No | built-in | Custom VCL snippet for vcl_backend_response |
@@ -314,6 +315,15 @@ Requires the [cloud-vinyl operator](https://github.com/bluedynamics/cloud-vinyl)
314
315
  | `monitoring` | `boolean` | No | `false` | Enable Prometheus metrics and ServiceMonitor |
315
316
  | `tolerations` | `VinylCacheToleration[]` | No | - | Node tolerations for Varnish pods |
316
317
 
318
+ **`VinylCacheStorage` fields:**
319
+
320
+ | Property | Type | Required | Description |
321
+ |----------|------|----------|-------------|
322
+ | `name` | `string` | Yes | Internal storage identifier (must match `^[a-zA-Z][a-zA-Z0-9_]*$`) |
323
+ | `type` | `'malloc' \| 'file'` | Yes | Storage backend type |
324
+ | `size` | `string` | Yes | Kubernetes resource quantity (e.g. `"1Gi"`, `"500M"`) |
325
+ | `path` | `string` | for `file` | Filesystem path for file-type storage |
326
+
317
327
  **Example:**
318
328
  ```typescript
319
329
  const cache = new PloneVinylCache(chart, 'cache', {
@@ -343,6 +353,19 @@ new PloneVinylCache(chart, 'cache', {
343
353
  });
344
354
  ```
345
355
 
356
+ **Storage sizing:**
357
+ ```typescript
358
+ new PloneVinylCache(chart, 'cache', {
359
+ plone: ploneInstance,
360
+ limitMemory: '2Gi',
361
+ storage: [
362
+ { name: 's0', type: 'malloc', size: '1500M' },
363
+ ],
364
+ });
365
+ ```
366
+
367
+ Without an explicit `storage` entry, varnishd runs with its stock default (~100 MB malloc) regardless of the container's memory limit. Size malloc storage below the pod's memory limit to leave headroom for varnishd overhead and transient allocations.
368
+
346
369
  **vs PloneHttpcache:**
347
370
  - `PloneHttpcache` deploys Varnish via mittwald Helm chart (self-contained, no operator needed)
348
371
  - `PloneVinylCache` creates a VinylCache CR managed by the cloud-vinyl operator (requires operator in cluster)
package/lib/httpcache.js CHANGED
@@ -118,5 +118,5 @@ class PloneHttpcache extends constructs_1.Construct {
118
118
  }
119
119
  exports.PloneHttpcache = PloneHttpcache;
120
120
  _a = JSII_RTTI_SYMBOL_1;
121
- PloneHttpcache[_a] = { fqn: "@bluedynamics/cdk8s-plone.PloneHttpcache", version: "0.1.30" };
121
+ PloneHttpcache[_a] = { fqn: "@bluedynamics/cdk8s-plone.PloneHttpcache", version: "0.1.32" };
122
122
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"httpcache.js","sourceRoot":"","sources":["../src/httpcache.ts"],"names":[],"mappings":"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","sourcesContent":["import * as fs from 'fs';\nimport * as path from 'path';\nimport { Helm } from 'cdk8s';\nimport { Construct } from 'constructs';\nimport { Plone } from './plone';\n\n/**\n * An environment variable to pass to the kube-httpcache container.\n */\nexport interface HttpcacheEnvVar {\n  /**\n   * The name of the environment variable.\n   */\n  readonly name: string;\n\n  /**\n   * The value of the environment variable.\n   */\n  readonly value: string;\n}\n\n/**\n * A Kubernetes toleration for the Varnish pods.\n */\nexport interface HttpcacheToleration {\n  /**\n   * The taint key to tolerate.\n   */\n  readonly key: string;\n\n  /**\n   * The operator (Equal or Exists).\n   * @default 'Equal'\n   */\n  readonly operator?: string;\n\n  /**\n   * The taint value to match (when operator is Equal).\n   * @default - no value\n   */\n  readonly value?: string;\n\n  /**\n   * The taint effect to tolerate (NoSchedule, PreferNoSchedule, NoExecute).\n   * @default - tolerate all effects\n   */\n  readonly effect?: string;\n}\n\n/**\n * Configuration options for PloneHttpcache (Varnish caching layer).\n */\nexport interface PloneHttpcacheOptions {\n  /**\n   * The Plone construct to attach the HTTP cache to.\n   * The cache will automatically connect to the backend and frontend services.\n   */\n  readonly plone: Plone;\n\n  /**\n   * Varnish VCL configuration as a string.\n   * If provided, this takes precedence over varnishVclFile.\n   * @default - loaded from varnishVclFile or default config file\n   */\n  readonly varnishVcl?: string;\n\n  /**\n   * Path to a Varnish VCL configuration file.\n   * If not provided, uses the default VCL file included in the library.\n   * @default - uses default config/varnish.tpl.vcl\n   */\n  readonly varnishVclFile?: string | undefined;\n\n  /**\n   * Name of an existing Kubernetes secret containing Varnish admin credentials.\n   * The secret should be created separately in the same namespace.\n   * @default - undefined (no existing secret)\n   */\n  readonly existingSecret?: string;\n\n  /**\n   * CPU limit for Varnish pods.\n   * @default '500m'\n   */\n  readonly limitCpu?: string;\n\n  /**\n   * Memory limit for Varnish pods.\n   * @default '500Mi'\n   */\n  readonly limitMemory?: string;\n\n  /**\n   * CPU request for Varnish pods.\n   * @default '100m'\n   */\n  readonly requestCpu?: string;\n\n  /**\n   * Memory request for Varnish pods.\n   * @default '100Mi'\n   */\n  readonly requestMemory?: string;\n\n  /**\n   * Enable Prometheus ServiceMonitor for metrics collection.\n   * Requires Prometheus Operator to be installed in the cluster.\n   * @default false\n   */\n  readonly servicemonitor?: boolean;\n\n  /**\n   * Enable the Prometheus exporter for Varnish metrics.\n   * When enabled, the exporter sidecar container will be deployed alongside Varnish.\n   * @default true\n   */\n  readonly exporterEnabled?: boolean;\n\n  /**\n   * Version of the kube-httpcache Helm chart to use.\n   * If not specified, the latest version from the repository will be used.\n   * @default undefined (latest)\n   */\n  readonly chartVersion?: string;\n\n  /**\n   * Version of the kube-httpcache Container Image to use.\n   * If not specified, the latest version from the repository will be used.\n   * @default undefined (chartVersion = with each chart release there is an image release too )\n   */\n  readonly appVersion?: string;\n\n  /**\n   * Number of Varnish pod replicas to run.\n   * @default 2\n   */\n  readonly replicas?: number;\n\n  /**\n   * Additional environment variables to pass to the kube-httpcache container.\n   * These are appended to the built-in env vars (BACKEND_SERVICE_NAME, etc.)\n   * and can be referenced in VCL templates using Go template syntax: {{ .Env.VAR_NAME }}\n   * @default - no additional env vars\n   */\n  readonly extraEnvVars?: HttpcacheEnvVar[];\n\n  /**\n   * Tolerations for the Varnish pods.\n   * Use this to allow scheduling on nodes with specific taints,\n   * e.g. nodes tainted with kubernetes.io/arch=amd64:NoSchedule.\n   * @default - no tolerations\n   */\n  readonly tolerations?: HttpcacheToleration[];\n}\n\n/**\n * PloneHttpcache construct for deploying Varnish HTTP caching layer.\n *\n * Uses the mittwald/kube-httpcache Helm chart to deploy Varnish as a\n * caching proxy in front of Plone backend and/or frontend services.\n *\n * The cache automatically connects to the Plone services and provides\n * HTTP cache invalidation capabilities.\n *\n * @example\n * const plone = new Plone(chart, 'plone');\n * const cache = new PloneHttpcache(chart, 'cache', {\n *   plone: plone,\n *   existingSecret: 'varnish-secret',\n * });\n */\nexport class PloneHttpcache extends Construct {\n  /**\n   * Name of the Varnish service created by the Helm chart.\n   * Use this to reference the cache service from ingress or other constructs.\n   */\n  public readonly httpcacheServiceName: string;\n\n  constructor(scope: Construct, id: string, options: PloneHttpcacheOptions) {\n    super(scope, id);\n    let varnishVcl: string;\n    if (!options.varnishVcl) {\n      let vclPath: string;\n      if (!options.varnishVclFile) {\n        vclPath = path.join(__dirname, 'config', 'varnish.tpl.vcl');\n      } else {\n        vclPath = options.varnishVclFile;\n      }\n      varnishVcl = fs.readFileSync(vclPath, 'utf8');\n    } else {\n      varnishVcl = options.varnishVcl;\n    }\n    const imageTag = options.appVersion ?? options.chartVersion;\n    const httpcache = new Helm(this, 'httpcache', {\n      // see https://github.com/mittwald/kube-httpcache/chart\n      repo: 'https://helm.mittwald.de',\n      chart: 'kube-httpcache',\n      version: options.chartVersion,\n      values: {\n        replicaCount: options.replicas ?? 2,\n        ...(imageTag && { image: { tag: imageTag } }),\n        cache: {\n          backendService: options.plone.backendServiceName,\n          frontendWatch: false,\n          backendWatch: true,\n          existingSecret: options.existingSecret ?? undefined,\n        },\n        // Workaround: upstream chart accepts backendPortName but never\n        // renders it as CLI arg. Pass via cacheExtraArgs instead.\n        cacheExtraArgs: '- -backend-portname=backend-http',\n        vclTemplate: varnishVcl,\n        extraEnvVars: [\n          { name: 'BACKEND_SERVICE_NAME', value: options.plone.backendServiceName },\n          { name: 'BACKEND_SERVICE_PORT', value: '8080' },\n          { name: 'BACKEND_SITE_ID', value: options.plone.siteId },\n          { name: 'FRONTEND_SERVICE_NAME', value: options.plone.frontendServiceName },\n          { name: 'FRONTEND_SERVICE_PORT', value: '3000' },\n          ...(options.extraEnvVars ?? []),\n        ],\n        // see https://github.com/mittwald/kube-httpcache/issues/253\n        nodeSelector: {\n          'kubernetes.io/arch': 'amd64',\n        },\n        ...(options.tolerations && { tolerations: options.tolerations }),\n        resources: {\n          limits: {\n            cpu: options.limitCpu || '500m',\n            memory: options.limitMemory || '500Mi',\n          },\n          requests: {\n            cpu: options.requestCpu || '100m',\n            memory: options.requestMemory || '100Mi',\n          },\n        },\n        rbac: {\n          enabled: true,\n        },\n        exporter: {\n          enabled: options.exporterEnabled ?? true,\n          resources: {\n            limits: {\n              cpu: '100m',\n              memory: '100Mi',\n            },\n            requests: {\n              cpu: '10m',\n              memory: '50Mi',\n            },\n          },\n        },\n        serviceMonitor: {\n          enabled: options.servicemonitor || false,\n          scrapeSignaller: options.servicemonitor || false,\n        },\n      },\n    });\n    const httpcacheService = httpcache.apiObjects.find((construct) => {\n      if ((construct.kind === 'Service') && (construct.metadata.name?.endsWith('kube-httpcache'))) {\n        return construct.name;\n      }\n      return undefined;\n    });\n    if (httpcacheService === undefined) {\n      throw new Error('Could not find httpcache service');\n    }\n    this.httpcacheServiceName = httpcacheService.name;\n  }\n}"]}
package/lib/index.d.ts CHANGED
@@ -1,3 +1,3 @@
1
1
  export { Plone, PloneOptions, PloneBaseOptions, PloneVariant, PloneSecurityContext, PloneCapabilities } from './plone';
2
2
  export { PloneHttpcache, PloneHttpcacheOptions, HttpcacheEnvVar, HttpcacheToleration } from './httpcache';
3
- export { PloneVinylCache, PloneVinylCacheOptions, VinylCacheToleration, VinylCacheBackend, VinylCacheBackendProbe } from './vinylcache';
3
+ export { PloneVinylCache, PloneVinylCacheOptions, VinylCacheToleration, VinylCacheBackend, VinylCacheBackendProbe, VinylCacheStorage } from './vinylcache';
package/lib/index.js CHANGED
@@ -8,4 +8,4 @@ var httpcache_1 = require("./httpcache");
8
8
  Object.defineProperty(exports, "PloneHttpcache", { enumerable: true, get: function () { return httpcache_1.PloneHttpcache; } });
9
9
  var vinylcache_1 = require("./vinylcache");
10
10
  Object.defineProperty(exports, "PloneVinylCache", { enumerable: true, get: function () { return vinylcache_1.PloneVinylCache; } });
11
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsaUNBQXVIO0FBQTlHLDhGQUFBLEtBQUssT0FBQTtBQUFrQyxxR0FBQSxZQUFZLE9BQUE7QUFDNUQseUNBQTBHO0FBQWpHLDJHQUFBLGNBQWMsT0FBQTtBQUN2QiwyQ0FBd0k7QUFBL0gsNkdBQUEsZUFBZSxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgUGxvbmUsIFBsb25lT3B0aW9ucywgUGxvbmVCYXNlT3B0aW9ucywgUGxvbmVWYXJpYW50LCBQbG9uZVNlY3VyaXR5Q29udGV4dCwgUGxvbmVDYXBhYmlsaXRpZXMgfSBmcm9tICcuL3Bsb25lJztcbmV4cG9ydCB7IFBsb25lSHR0cGNhY2hlLCBQbG9uZUh0dHBjYWNoZU9wdGlvbnMsIEh0dHBjYWNoZUVudlZhciwgSHR0cGNhY2hlVG9sZXJhdGlvbiB9IGZyb20gJy4vaHR0cGNhY2hlJztcbmV4cG9ydCB7IFBsb25lVmlueWxDYWNoZSwgUGxvbmVWaW55bENhY2hlT3B0aW9ucywgVmlueWxDYWNoZVRvbGVyYXRpb24sIFZpbnlsQ2FjaGVCYWNrZW5kLCBWaW55bENhY2hlQmFja2VuZFByb2JlIH0gZnJvbSAnLi92aW55bGNhY2hlJztcbiJdfQ==
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsaUNBQXVIO0FBQTlHLDhGQUFBLEtBQUssT0FBQTtBQUFrQyxxR0FBQSxZQUFZLE9BQUE7QUFDNUQseUNBQTBHO0FBQWpHLDJHQUFBLGNBQWMsT0FBQTtBQUN2QiwyQ0FBMko7QUFBbEosNkdBQUEsZUFBZSxPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgUGxvbmUsIFBsb25lT3B0aW9ucywgUGxvbmVCYXNlT3B0aW9ucywgUGxvbmVWYXJpYW50LCBQbG9uZVNlY3VyaXR5Q29udGV4dCwgUGxvbmVDYXBhYmlsaXRpZXMgfSBmcm9tICcuL3Bsb25lJztcbmV4cG9ydCB7IFBsb25lSHR0cGNhY2hlLCBQbG9uZUh0dHBjYWNoZU9wdGlvbnMsIEh0dHBjYWNoZUVudlZhciwgSHR0cGNhY2hlVG9sZXJhdGlvbiB9IGZyb20gJy4vaHR0cGNhY2hlJztcbmV4cG9ydCB7IFBsb25lVmlueWxDYWNoZSwgUGxvbmVWaW55bENhY2hlT3B0aW9ucywgVmlueWxDYWNoZVRvbGVyYXRpb24sIFZpbnlsQ2FjaGVCYWNrZW5kLCBWaW55bENhY2hlQmFja2VuZFByb2JlLCBWaW55bENhY2hlU3RvcmFnZSB9IGZyb20gJy4vdmlueWxjYWNoZSc7XG4iXX0=
package/lib/plone.js CHANGED
@@ -262,5 +262,5 @@ class Plone extends constructs_1.Construct {
262
262
  }
263
263
  exports.Plone = Plone;
264
264
  _a = JSII_RTTI_SYMBOL_1;
265
- Plone[_a] = { fqn: "@bluedynamics/cdk8s-plone.Plone", version: "0.1.30" };
265
+ Plone[_a] = { fqn: "@bluedynamics/cdk8s-plone.Plone", version: "0.1.32" };
266
266
  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plone.js","sourceRoot":"","sources":["../src/plone.ts"],"names":[],"mappings":"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","sourcesContent":["import { Names } from 'cdk8s';\nimport * as kplus from 'cdk8s-plus-30';\nimport { Construct } from 'constructs';\nimport { PloneDeployment, PloneDeploymentOptions } from './deployment';\nimport { IntOrString } from './imports/k8s';\nimport * as k8s from './imports/k8s';\nimport { ServiceMonitor, ServiceMonitorSpecEndpointsTargetPort } from './imports/monitoring.coreos.com';\nimport { PloneService } from './service';\n\n/**\n * Linux capabilities to add or drop on a container.\n */\nexport interface PloneCapabilities {\n  /**\n   * Capabilities to add (e.g. 'SYS_PTRACE', 'NET_ADMIN').\n   * @default - no capabilities added\n   */\n  readonly add?: string[];\n\n  /**\n   * Capabilities to drop (e.g. 'ALL', 'NET_RAW').\n   * @default - no capabilities dropped\n   */\n  readonly drop?: string[];\n}\n\n/**\n * Security context for a Plone container.\n * Controls privilege and access settings.\n */\nexport interface PloneSecurityContext {\n  /**\n   * Linux capabilities to add or drop.\n   * @example { add: ['SYS_PTRACE'] }\n   * @default - no capability changes\n   */\n  readonly capabilities?: PloneCapabilities;\n\n  /**\n   * Run the container as a specific user ID.\n   * @default - container default\n   */\n  readonly runAsUser?: number;\n\n  /**\n   * Run the container as a specific group ID.\n   * @default - container default\n   */\n  readonly runAsGroup?: number;\n\n  /**\n   * Require the container to run as non-root.\n   * @default - undefined\n   */\n  readonly runAsNonRoot?: boolean;\n\n  /**\n   * Mount the root filesystem as read-only.\n   * @default - undefined\n   */\n  readonly readOnlyRootFilesystem?: boolean;\n\n  /**\n   * Allow privilege escalation for the container process.\n   * @default - undefined\n   */\n  readonly allowPrivilegeEscalation?: boolean;\n\n  /**\n   * Run the container in privileged mode.\n   * @default - undefined\n   */\n  readonly privileged?: boolean;\n}\n\n/**\n * Base options for Plone backend or frontend configuration.\n * These options control container image, replica count, resource limits,\n * environment variables, and health probes.\n */\nexport interface PloneBaseOptions {\n  /**\n   * Container image to use for the deployment.\n   * @example 'plone/plone-backend:6.0.10' or 'plone/plone-frontend:16.0.0'\n   * @default - 'plone/plone-backend:latest' for backend, 'plone/plone-frontend:latest' for frontend\n   */\n  readonly image?: string;\n\n  /**\n   * Image pull policy for the container.\n   * @default 'IfNotPresent'\n   */\n  readonly imagePullPolicy?: string;\n\n  /**\n   * Number of pod replicas to run.\n   * @default 2\n   */\n  readonly replicas?: number;\n\n  /**\n   * Maximum number of pods that can be unavailable during updates.\n   * Can be an absolute number (e.g., 1) or a percentage (e.g., '50%').\n   * Used in PodDisruptionBudget if specified.\n   * @default - undefined (not set)\n   */\n  readonly maxUnavailable?: number | string;\n\n  /**\n   * Minimum number of pods that must be available during updates.\n   * Can be an absolute number (e.g., 1) or a percentage (e.g., '50%').\n   * Used in PodDisruptionBudget if specified.\n   * @default - undefined (not set)\n   */\n  readonly minAvailable?: number | string;\n\n  /**\n   * CPU limit for the container.\n   * @example '500m' or '1' or '2000m'\n   * @default '500m' for both backend and frontend\n   */\n  readonly limitCpu?: string;\n\n  /**\n   * Memory limit for the container.\n   * @example '512Mi' or '1Gi'\n   * @default '512Mi' for backend, '1Gi' for frontend\n   */\n  readonly limitMemory?: string;\n\n  /**\n   * CPU request for the container.\n   * @example '200m' or '0.5'\n   * @default '200m'\n   */\n  readonly requestCpu?: string;\n\n  /**\n   * Memory request for the container.\n   * @example '256Mi' or '512Mi'\n   * @default '256Mi'\n   */\n  readonly requestMemory?: string;\n\n  /**\n   * Environment variables to set in the container.\n   * Use cdk8s-plus-30 Env class to define variables and sources.\n   * @default - undefined (no additional environment variables)\n   */\n  readonly environment?: kplus.Env;\n\n  /**\n   * Enable readiness probe for the container.\n   * Readiness probes determine when a container is ready to accept traffic.\n   * @default true\n   */\n  readonly readinessEnabled?: boolean;\n\n  /**\n   * Number of seconds after container start before readiness probe is initiated.\n   * @default 10\n   */\n  readonly readinessInitialDelaySeconds?: number;\n\n  /**\n   * Number of seconds after which the readiness probe times out.\n   * @default 15\n   */\n  readonly readinessTimeoutSeconds?: number;\n\n  /**\n   * How often (in seconds) to perform the readiness probe.\n   * @default 10\n   */\n  readonly readinessPeriodSeconds?: number;\n\n  /**\n   * Minimum consecutive successes for the readiness probe to be considered successful.\n   * @default 1\n   */\n  readonly readinessSuccessThreshold?: number;\n\n  /**\n   * Minimum consecutive failures for the readiness probe to be considered failed.\n   * @default 3\n   */\n  readonly readinessFailureThreshold?: number;\n\n  /**\n   * Enable liveness probe for the container.\n   * Liveness probes determine when to restart a container.\n   * Recommended: true for frontend, false for backend (Zope has its own recovery).\n   * @default false\n   */\n  readonly livenessEnabled?: boolean;\n\n  /**\n   * Number of seconds after container start before liveness probe is initiated.\n   * @default 30\n   */\n  readonly livenessInitialDelaySeconds?: number;\n\n  /**\n   * Number of seconds after which the liveness probe times out.\n   * @default 5\n   */\n  readonly livenessTimeoutSeconds?: number;\n\n  /**\n   * How often (in seconds) to perform the liveness probe.\n   * @default 10\n   */\n  readonly livenessPeriodSeconds?: number;\n\n  /**\n   * Minimum consecutive successes for the liveness probe to be considered successful.\n   * @default 1\n   */\n  readonly livenessSuccessThreshold?: number;\n\n  /**\n   * Minimum consecutive failures for the liveness probe to be considered failed.\n   * @default 3\n   */\n  readonly livenessFailureThreshold?: number;\n\n  /**\n   * Annotations to add to the Deployment metadata.\n   * @example { 'deployment.kubernetes.io/revision': '1' }\n   * @default - no additional annotations\n   */\n  readonly annotations?: { [name: string]: string };\n\n  /**\n   * Annotations to add to the Pod template metadata.\n   * Common for Prometheus, Istio, backup policies, etc.\n   * @example { 'prometheus.io/scrape': 'true', 'prometheus.io/port': '8080' }\n   * @default - no additional annotations\n   */\n  readonly podAnnotations?: { [name: string]: string };\n\n  /**\n   * Annotations to add to the Service metadata.\n   * Common for external-dns, load balancers, service mesh, etc.\n   * @example { 'external-dns.alpha.kubernetes.io/hostname': 'plone.example.com' }\n   * @default - no additional annotations\n   */\n  readonly serviceAnnotations?: { [name: string]: string };\n\n  /**\n   * Enable Prometheus ServiceMonitor for metrics collection.\n   * Requires Prometheus Operator to be installed in the cluster.\n   * When enabled, a ServiceMonitor resource will be created to scrape metrics.\n   * @default false\n   */\n  readonly servicemonitor?: boolean;\n\n  /**\n   * Port name or number to scrape metrics from.\n   * Only used when servicemonitor is enabled.\n   * @default - uses the main service port\n   */\n  readonly metricsPort?: string | number;\n\n  /**\n   * Path to scrape metrics from.\n   * Only used when servicemonitor is enabled.\n   * @default '/metrics'\n   */\n  readonly metricsPath?: string;\n\n  /**\n   * Node selector labels for pod scheduling.\n   * Use to constrain pods to nodes with matching labels, e.g. for region affinity.\n   * @example { 'topology.kubernetes.io/region': 'fsn1' }\n   * @default - no node selector\n   */\n  readonly nodeSelector?: { [key: string]: string };\n\n  /**\n   * Security context for the container.\n   * Use to set capabilities, run as non-root, read-only filesystem, etc.\n   * @example { capabilities: { add: ['SYS_PTRACE'] } }\n   * @default - no security context\n   */\n  readonly securityContext?: PloneSecurityContext;\n}\n/**\n * Plone deployment variants.\n */\nexport enum PloneVariant {\n  /**\n   * Volto variant: ReactJS frontend (Volto) with REST API backend.\n   * Deploys both frontend and backend services.\n   */\n  VOLTO = 'volto',\n\n  /**\n   * Classic UI variant: Traditional Plone with server-side rendering.\n   * Deploys only the backend service.\n   */\n  CLASSICUI = 'classicui',\n}\n\n/**\n * Main configuration options for Plone deployment.\n */\nexport interface PloneOptions {\n  /**\n   * Version string for labeling the deployment.\n   * This is used in Kubernetes labels and doesn't affect the actual image versions.\n   * @default 'undefined'\n   */\n  readonly version?: string;\n\n  /**\n   * Plone site ID in the ZODB.\n   * This is used to construct the internal API path for Volto frontend.\n   * @default 'Plone'\n   */\n  readonly siteId?: string;\n\n  /**\n   * Plone deployment variant to use.\n   * @default PloneVariant.VOLTO\n   */\n  readonly variant?: PloneVariant;\n\n  /**\n   * Backend (Plone API) configuration.\n   * @default {} (uses default values from PloneBaseOptions)\n   */\n  readonly backend?: PloneBaseOptions;\n\n  /**\n   * Frontend (Volto) configuration.\n   * Only used when variant is PloneVariant.VOLTO.\n   * @default {} (uses default values from PloneBaseOptions)\n   */\n  readonly frontend?: PloneBaseOptions;\n\n  /**\n   * Names of Kubernetes secrets to use for pulling private container images.\n   * These secrets must exist in the same namespace as the deployment.\n   * @example ['my-registry-secret']\n   * @default [] (no image pull secrets)\n   */\n  readonly imagePullSecrets?: string[];\n}\n\n/**\n * Plone construct for deploying Plone CMS to Kubernetes.\n *\n * This construct creates all necessary Kubernetes resources for running Plone:\n * - Deployment(s) for backend (and optionally frontend)\n * - Service(s) for network access\n * - Optional PodDisruptionBudget for high availability\n *\n * Supports two deployment variants:\n * - VOLTO: Modern React frontend with REST API backend (default)\n * - CLASSICUI: Traditional server-side rendered Plone\n *\n * @example\n * new Plone(chart, 'my-plone', {\n *   variant: PloneVariant.VOLTO,\n *   backend: {\n *     image: 'plone/plone-backend:6.0.10',\n *     replicas: 3,\n *   },\n *   frontend: {\n *     image: 'plone/plone-frontend:16.0.0',\n *   },\n * });\n */\nexport class Plone extends Construct {\n  /**\n   * Name of the backend Kubernetes service.\n   * Use this to reference the backend service from other constructs.\n   */\n  public readonly backendServiceName: string;\n\n  /**\n   * Name of the frontend Kubernetes service.\n   * Only set when variant is VOLTO, otherwise undefined.\n   */\n  public readonly frontendServiceName: string | undefined;\n\n  /**\n   * The deployment variant being used (VOLTO or CLASSICUI).\n   */\n  public readonly variant: PloneVariant;\n\n  /**\n   * The Plone site ID in ZODB.\n   */\n  public readonly siteId: string;\n\n  constructor(scope: Construct, id: string, options: PloneOptions = {}) {\n    super(scope, id);\n    this.frontendServiceName = undefined;\n    this.siteId = options.siteId ?? 'Plone';\n    this.variant = options.variant ?? PloneVariant.VOLTO;\n\n    // ------------------------------------------------------------------------\n    // Backend\n    const backend = options.backend ?? {};\n    const backendLabels = {\n      'app.kubernetes.io/name': 'plone-backend',\n      'app.kubernetes.io/component': 'backend',\n      'app.kubernetes.io/version': options.version ?? 'undefined',\n    };\n    const backendPort = 8080;\n\n    // Options\n    var backendOptions: PloneDeploymentOptions = {\n      labels: backendLabels,\n      image: {\n        image: backend.image ?? 'plone/plone-backend:latest',\n        imagePullSecrets: options.imagePullSecrets ?? [],\n        imagePullPolicy: backend.imagePullPolicy ?? 'IfNotPresent',\n      },\n      replicas: backend.replicas,\n      limitCpu: backend.limitCpu ?? '500m',\n      limitMemory: backend.limitMemory ?? '512Mi',\n      requestCpu: backend.requestCpu ?? '200m',\n      requestMemory: backend.requestMemory ?? '256Mi',\n      pdb: {\n        maxUnavailable: backend.maxUnavailable ?? undefined,\n        minAvailable: backend.minAvailable ?? undefined,\n      },\n      port: backendPort,\n      environment: backend.environment,\n      annotations: backend.annotations,\n      podAnnotations: backend.podAnnotations,\n      nodeSelector: backend.nodeSelector,\n      securityContext: backend.securityContext as k8s.SecurityContext,\n    };\n\n    // Probing\n    const backendActionHttpGet: k8s.HttpGetAction = {\n      path: '/',\n      port: IntOrString.fromNumber(backendPort),\n    };\n    if (backend.livenessEnabled ?? false) {\n      backendOptions.livenessProbe = {\n        httpGet: backendActionHttpGet,\n        initialDelaySeconds: backend.livenessInitialDelaySeconds ?? 30,\n        timeoutSeconds: backend.livenessTimeoutSeconds ?? 5,\n        periodSeconds: backend.livenessPeriodSeconds ?? 10,\n        successThreshold: backend.livenessSuccessThreshold ?? 1,\n        failureThreshold: backend.livenessFailureThreshold ?? 3,\n      };\n    }\n    if (backend.readinessEnabled ?? true) {\n      backendOptions.readinessProbe = {\n        httpGet: backendActionHttpGet,\n        initialDelaySeconds: backend.readinessInitialDelaySeconds ?? 10,\n        timeoutSeconds: backend.readinessTimeoutSeconds ?? 15,\n        periodSeconds: backend.readinessPeriodSeconds ?? 10,\n        successThreshold: backend.readinessSuccessThreshold ?? 1,\n        failureThreshold: backend.readinessFailureThreshold ?? 3,\n      };\n    }\n    // Deployment\n    var backendDeployment = new PloneDeployment(this, 'backend', backendOptions);\n\n    // Service\n    const backendService = new PloneService(backendDeployment, 'service', {\n      labels: {\n        'app.kubernetes.io/name': 'plone-backend-service',\n        'app.kubernetes.io/component': 'service',\n      },\n      targetPort: backendPort,\n      selectorLabel: { app: Names.toLabelValue(backendDeployment) },\n      portName: 'backend-http',\n      annotations: backend.serviceAnnotations,\n    });\n    this.backendServiceName = backendService.name;\n\n    // ServiceMonitor for backend\n    if (backend.servicemonitor) {\n      this.createServiceMonitor(backendService, 'backend', {\n        port: backend.metricsPort ?? 'backend-http',\n        path: backend.metricsPath ?? '/metrics',\n      });\n    }\n\n    // ------------------------------------------------------------------------\n    // Frontend\n    if (this.variant == PloneVariant.VOLTO) {\n      const frontend = options.frontend ?? {};\n      const frontendPort = 3000;\n      const frontendLabels = {\n        'app.kubernetes.io/name': 'plone-frontend',\n        'app.kubernetes.io/component': 'frontend',\n        'app.kubernetes.io/version': options.version ?? 'undefined',\n      };\n\n      // Environment for RAZZLE\n      var frontendEnvironment = frontend.environment ?? new kplus.Env([], {});\n      if (frontendEnvironment.variables.RAZZLE_INTERNAL_API_PATH === undefined) {\n        // connect with backend service\n        frontendEnvironment?.addVariable('RAZZLE_INTERNAL_API_PATH', kplus.EnvValue.fromValue(`http://${backendService.name}:${backendPort}/${this.siteId}`));\n      }\n\n      // Options\n      var frontendOptions: PloneDeploymentOptions = {\n        labels: frontendLabels,\n        image: {\n          image: frontend.image ?? 'plone/plone-frontend:latest',\n          imagePullSecrets: options.imagePullSecrets ?? [],\n          imagePullPolicy: frontend.imagePullPolicy ?? 'IfNotPresent',\n        },\n        replicas: frontend.replicas,\n        limitCpu: frontend.limitCpu ?? '500m',\n        limitMemory: frontend.limitMemory ?? '1Gi',\n        requestCpu: frontend.requestCpu ?? '200m',\n        requestMemory: frontend.requestMemory ?? '256Mi',\n\n        pdb: {\n          maxUnavailable: frontend.maxUnavailable ?? undefined,\n          minAvailable: frontend.minAvailable ?? undefined,\n        },\n        port: frontendPort,\n        environment: frontendEnvironment,\n        annotations: frontend.annotations,\n        podAnnotations: frontend.podAnnotations,\n        nodeSelector: frontend.nodeSelector,\n        securityContext: frontend.securityContext as k8s.SecurityContext,\n      };\n\n      // Probing\n      const frontendActionHttpGet: k8s.HttpGetAction = {\n        path: '/',\n        port: IntOrString.fromNumber(frontendPort),\n      };\n      if (frontend.livenessEnabled ?? false) {\n        frontendOptions.livenessProbe = {\n          httpGet: frontendActionHttpGet,\n          initialDelaySeconds: frontend.livenessInitialDelaySeconds ?? 30,\n          timeoutSeconds: frontend.livenessTimeoutSeconds ?? 5,\n          periodSeconds: frontend.livenessPeriodSeconds ?? 10,\n          successThreshold: frontend.livenessSuccessThreshold ?? 1,\n          failureThreshold: frontend.livenessFailureThreshold ?? 3,\n        };\n      }\n      if (frontend.readinessEnabled ?? true) {\n        frontendOptions.readinessProbe = {\n          httpGet: frontendActionHttpGet,\n          initialDelaySeconds: frontend.readinessInitialDelaySeconds ?? 10,\n          timeoutSeconds: frontend.readinessTimeoutSeconds ?? 15,\n          periodSeconds: frontend.readinessPeriodSeconds ?? 10,\n          successThreshold: frontend.readinessSuccessThreshold ?? 1,\n          failureThreshold: frontend.readinessFailureThreshold ?? 3,\n        };\n      }\n\n      // Deployment\n      const frontendDeployment = new PloneDeployment(this, 'frontend', frontendOptions);\n\n      // Service\n      const frontendService = new PloneService(frontendDeployment, 'service', {\n        labels: {\n          'app.kubernetes.io/name': 'plone-frontend-service',\n          'app.kubernetes.io/component': 'service',\n        },\n        targetPort: frontendPort,\n        selectorLabel: { app: Names.toLabelValue(frontendDeployment) },\n        portName: 'frontend-http',\n        annotations: frontend.serviceAnnotations,\n      });\n      this.frontendServiceName = frontendService.name;\n\n      // ServiceMonitor for frontend\n      if (frontend.servicemonitor) {\n        this.createServiceMonitor(frontendService, 'frontend', {\n          port: frontend.metricsPort ?? 'frontend-http',\n          path: frontend.metricsPath ?? '/metrics',\n        });\n      }\n    }\n  }\n\n  /**\n   * Creates a Prometheus ServiceMonitor resource for the given service.\n   * @param service The PloneService to monitor\n   * @param id Unique identifier for the ServiceMonitor\n   * @param config Configuration for the ServiceMonitor\n   */\n  private createServiceMonitor(\n    service: PloneService,\n    id: string,\n    config: { port: string | number; path: string },\n  ): void {\n    const portName = typeof config.port === 'string' ? config.port : undefined;\n    const portNumber = typeof config.port === 'number'\n      ? ServiceMonitorSpecEndpointsTargetPort.fromNumber(config.port)\n      : undefined;\n\n    new ServiceMonitor(this, `servicemonitor-${id}`, {\n      metadata: {\n        name: `${service.name}-monitor`,\n        labels: {\n          'app.kubernetes.io/name': service.name,\n          'app.kubernetes.io/component': 'servicemonitor',\n        },\n      },\n      spec: {\n        selector: {\n          matchLabels: {\n            'app.kubernetes.io/name': service.labels['app.kubernetes.io/name'],\n          },\n        },\n        endpoints: [\n          {\n            ...(portName && { port: portName }),\n            ...(portNumber && { targetPort: portNumber }),\n            path: config.path,\n            interval: '30s',\n          },\n        ],\n      },\n    });\n  }\n}"]}
@@ -62,6 +62,38 @@ export interface VinylCacheBackend {
62
62
  */
63
63
  readonly weight?: number;
64
64
  }
65
+ /**
66
+ * A Varnish storage backend configuration.
67
+ *
68
+ * Maps to `spec.storage[]` on the VinylCache CRD. The operator emits one
69
+ * `-s <name>=<type>,<options>` argument per entry to varnishd.
70
+ *
71
+ * Without any storage entry the operator falls back to the varnishd default
72
+ * (~100 MB malloc), which is almost always too small for real workloads.
73
+ */
74
+ export interface VinylCacheStorage {
75
+ /**
76
+ * Internal storage identifier used in the varnishd `-s` argument.
77
+ * Must be unique within the VinylCache and match `^[a-zA-Z][a-zA-Z0-9_]*$`.
78
+ */
79
+ readonly name: string;
80
+ /**
81
+ * Storage backend type. Only "malloc" and "file" are permitted by the
82
+ * admission webhook.
83
+ */
84
+ readonly type: 'malloc' | 'file';
85
+ /**
86
+ * Storage allocation as a Kubernetes resource quantity (e.g. "1Gi", "500M").
87
+ * Required for malloc; required for file.
88
+ * @default - required for both malloc and file
89
+ */
90
+ readonly size?: string;
91
+ /**
92
+ * Filesystem path for file-type storage. Ignored for malloc.
93
+ * @default - required for type "file"
94
+ */
95
+ readonly path?: string;
96
+ }
65
97
  /**
66
98
  * A Kubernetes toleration for the Varnish pods.
67
99
  */
@@ -131,11 +163,54 @@ export interface PloneVinylCacheOptions {
131
163
  * @default '512Mi'
132
164
  */
133
165
  readonly limitMemory?: string;
166
+ /**
167
+ * Varnish storage backends (`spec.storage`).
168
+ *
169
+ * Each entry becomes a `-s <name>=<type>,<options>` argument to varnishd.
170
+ * If omitted, the operator ships varnishd with its stock default (~100 MB
171
+ * malloc) — almost always too small. Set an explicit malloc size at least
172
+ * matching the pod's memory request to use the allocated memory for caching.
173
+ *
174
+ * @default - no storage configured; operator uses varnishd default (~100MB malloc)
175
+ * @example
176
+ * storage: [{ name: 's0', type: 'malloc', size: '1Gi' }]
177
+ */
178
+ readonly storage?: VinylCacheStorage[];
134
179
  /**
135
180
  * Director type for load distribution.
136
181
  * @default 'shard'
137
182
  */
138
183
  readonly director?: string;
184
+ /**
185
+ * Shard director: what value is hashed for shard selection.
186
+ * "HASH" uses the Varnish hash (default); "URL" uses the request URL.
187
+ * Only applied when director is "shard".
188
+ * Requires cloud-vinyl operator >= 0.4.2.
189
+ * @default - operator default ("HASH")
190
+ */
191
+ readonly shardBy?: string;
192
+ /**
193
+ * Shard director: which backends the director considers when selecting a shard.
194
+ * "CHOSEN" (default) only considers the chosen backend healthy; "ALL" requires all
195
+ * backends to be healthy.
196
+ * Only applied when director is "shard".
197
+ * Requires cloud-vinyl operator >= 0.4.2.
198
+ * @default - operator default ("CHOSEN")
199
+ */
200
+ readonly shardHealthy?: string;
201
+ /**
202
+ * Shard director: time after adding a backend before it receives its full share of
203
+ * traffic, preventing thundering-herd.
204
+ * Only applied when director is "shard".
205
+ * @default - operator default ("30s")
206
+ */
207
+ readonly shardRampup?: string;
208
+ /**
209
+ * Shard director: number of Ketama replicas per backend in the hash ring.
210
+ * Only applied when director is "shard".
211
+ * @default - operator default (67)
212
+ */
213
+ readonly shardReplicas?: number;
139
214
  /**
140
215
  * Custom VCL snippet for vcl_recv subroutine.
141
216
  * Replaces the default Plone recv snippet.