@bluedynamics/cdk8s-plone 0.0.81 → 0.1.1
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/.jsii +356 -118
- package/API.md +433 -68
- package/README.md +200 -147
- package/lib/deployment.d.ts +76 -40
- package/lib/deployment.js +13 -2
- package/lib/httpcache.d.ts +76 -9
- package/lib/httpcache.js +21 -4
- package/lib/pdb.d.ts +22 -7
- package/lib/pdb.js +8 -1
- package/lib/plone.d.ts +215 -2
- package/lib/plone.js +49 -6
- package/lib/service.d.ts +26 -0
- package/lib/service.js +10 -2
- package/package.json +10 -10
package/README.md
CHANGED
|
@@ -1,181 +1,203 @@
|
|
|
1
|
-
#
|
|
1
|
+
# CDK8S Plone
|
|
2
2
|
|
|
3
|
-
|
|
3
|
+
A CDK8S library for deploying Plone CMS to Kubernetes.
|
|
4
4
|
|
|
5
|
-
|
|
6
|
-
- Backend (as API with `plone.volto` or as Classic-UI)
|
|
7
|
-
- Frontend (Plone-Volto, a ReactJS based user interface)
|
|
8
|
-
- Varnish using kube-httpcache. It includes a way to invalidate varnish cluster (optional)
|
|
5
|
+
This library provides constructs to bootstrap a Plone deployment on a Kubernetes cluster using the [CDK8S](https://cdk8s.io) framework.
|
|
9
6
|
|
|
7
|
+
## Features
|
|
10
8
|
|
|
11
|
-
|
|
9
|
+
- **Backend**: Plone backend (API with `plone.volto` or Classic-UI)
|
|
10
|
+
- **Frontend**: Plone Volto (modern React-based user interface)
|
|
11
|
+
- **Varnish Caching**: Optional HTTP caching layer using [kube-httpcache](https://github.com/mittwald/kube-httpcache) with cluster-wide cache invalidation
|
|
12
|
+
- **High Availability**: Configurable replicas with PodDisruptionBudgets
|
|
13
|
+
- **Multi-language Support**: Published to npm (TypeScript/JavaScript) and PyPI (Python)
|
|
12
14
|
|
|
13
|
-
|
|
15
|
+
|
|
16
|
+
## Installation
|
|
17
|
+
|
|
18
|
+
### TypeScript/JavaScript
|
|
19
|
+
|
|
20
|
+
Create a new CDK8S project (or use an existing one):
|
|
14
21
|
|
|
15
22
|
```bash
|
|
16
23
|
cdk8s init typescript-app
|
|
17
24
|
```
|
|
18
25
|
|
|
19
|
-
|
|
26
|
+
Install the library:
|
|
20
27
|
|
|
21
|
-
```
|
|
22
|
-
|
|
23
|
-
"dependencies": {
|
|
24
|
-
"@bluedynamics/cdk8s-plone": "*"
|
|
25
|
-
}
|
|
26
|
-
}
|
|
28
|
+
```bash
|
|
29
|
+
npm install @bluedynamics/cdk8s-plone
|
|
27
30
|
```
|
|
28
|
-
|
|
31
|
+
|
|
32
|
+
Package: [@bluedynamics/cdk8s-plone](https://www.npmjs.com/package/@bluedynamics/cdk8s-plone)
|
|
29
33
|
|
|
30
34
|
### Python
|
|
31
35
|
|
|
32
|
-
|
|
36
|
+
Create a new CDK8S project:
|
|
33
37
|
|
|
34
38
|
```bash
|
|
35
39
|
cdk8s init python-app
|
|
36
40
|
```
|
|
37
41
|
|
|
38
|
-
|
|
42
|
+
Install the library:
|
|
39
43
|
|
|
44
|
+
```bash
|
|
45
|
+
pip install cdk8s-plone
|
|
46
|
+
```
|
|
40
47
|
|
|
41
|
-
|
|
48
|
+
Package: [cdk8s-plone](https://pypi.org/project/cdk8s-plone/)
|
|
42
49
|
|
|
43
|
-
With `cdk8s-cli` installed, create a new project:
|
|
44
50
|
|
|
45
|
-
|
|
46
|
-
|
|
51
|
+
## Quick Start
|
|
52
|
+
|
|
53
|
+
### Basic Plone Deployment
|
|
54
|
+
|
|
55
|
+
```typescript
|
|
56
|
+
import { App, Chart } from 'cdk8s';
|
|
57
|
+
import { Plone, PloneVariant } from '@bluedynamics/cdk8s-plone';
|
|
58
|
+
|
|
59
|
+
const app = new App();
|
|
60
|
+
const chart = new Chart(app, 'PloneDeployment');
|
|
61
|
+
|
|
62
|
+
new Plone(chart, 'my-plone', {
|
|
63
|
+
variant: PloneVariant.VOLTO,
|
|
64
|
+
backend: {
|
|
65
|
+
image: 'plone/plone-backend:6.0.10',
|
|
66
|
+
replicas: 3,
|
|
67
|
+
},
|
|
68
|
+
frontend: {
|
|
69
|
+
image: 'plone/plone-frontend:16.0.0',
|
|
70
|
+
replicas: 2,
|
|
71
|
+
},
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
app.synth();
|
|
47
75
|
```
|
|
48
76
|
|
|
49
|
-
|
|
77
|
+
### With Varnish HTTP Cache
|
|
50
78
|
|
|
51
79
|
```typescript
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
80
|
+
import { PloneHttpcache } from '@bluedynamics/cdk8s-plone';
|
|
81
|
+
|
|
82
|
+
const plone = new Plone(chart, 'my-plone', {
|
|
83
|
+
variant: PloneVariant.VOLTO,
|
|
84
|
+
backend: { image: 'plone/plone-backend:6.0.10' },
|
|
85
|
+
frontend: { image: 'plone/plone-frontend:16.0.0' },
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
new PloneHttpcache(chart, 'cache', {
|
|
89
|
+
plone: plone,
|
|
90
|
+
existingSecret: 'varnish-secret',
|
|
91
|
+
replicas: 2,
|
|
92
|
+
});
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
### Generate Kubernetes Manifests
|
|
96
|
+
|
|
97
|
+
```bash
|
|
98
|
+
cdk8s synth
|
|
60
99
|
```
|
|
61
100
|
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
- `
|
|
88
|
-
|
|
89
|
-
- `
|
|
90
|
-
- `
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
- `
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
- `
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
- `
|
|
107
|
-
- `
|
|
108
|
-
- `
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
- `
|
|
117
|
-
|
|
118
|
-
- `
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
- `
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
- `
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
-
|
|
134
|
-
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
- `
|
|
144
|
-
- `
|
|
145
|
-
|
|
146
|
-
- `
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
- `
|
|
153
|
-
- Plone chart
|
|
154
|
-
- `varnishVcl`{string}:
|
|
155
|
-
- varnishfile
|
|
156
|
-
- per default `varnishVclFile` should be used
|
|
157
|
-
- `varnishVclFile`(string):
|
|
158
|
-
- File in config folder
|
|
159
|
-
- `existingSecret`(string)
|
|
160
|
-
- `limitCpu`(string)
|
|
161
|
-
- `limitMemory`(string)
|
|
162
|
-
- `requestCpu`(string)
|
|
163
|
-
- `requestMemory`(string)
|
|
164
|
-
- `servicemonitor`(string)
|
|
165
|
-
- default `false` used for metrics
|
|
166
|
-
|
|
167
|
-
#### PloneHttpcache
|
|
168
|
-
*class*
|
|
169
|
-
|
|
170
|
-
uses helmchart [kube-httpcache](https://github.com/mittwald/kube-httpcache) and builds the `PloneHttpCache` Construct
|
|
171
|
-
- `scope`(Construct)
|
|
172
|
-
- `id`(string)
|
|
173
|
-
- `options`(PloneHttpcacheOptions)
|
|
101
|
+
The manifests are stored in the `dist/` directory.
|
|
102
|
+
|
|
103
|
+
For a complete example, see the [example project](https://github.com/bluedynamics/cdk8s-plone-example).
|
|
104
|
+
|
|
105
|
+
## Prerequisites
|
|
106
|
+
|
|
107
|
+
- **kubectl** - Command-line tool for deploying Kubernetes manifests. [Install kubectl](https://kubernetes.io/docs/tasks/tools/#kubectl)
|
|
108
|
+
- **Helm** (optional) - Only needed if generating Helm charts. [Install Helm](https://helm.sh/docs/intro/install/)
|
|
109
|
+
- **Node.js** - For TypeScript/JavaScript development
|
|
110
|
+
- **Python 3.8+** - For Python development
|
|
111
|
+
|
|
112
|
+
|
|
113
|
+
## API Documentation
|
|
114
|
+
|
|
115
|
+
For complete API documentation, see [API.md](./API.md).
|
|
116
|
+
|
|
117
|
+
### Key Constructs
|
|
118
|
+
|
|
119
|
+
#### `Plone`
|
|
120
|
+
|
|
121
|
+
Main construct for deploying Plone CMS. Supports two variants:
|
|
122
|
+
- **VOLTO**: Modern React frontend with REST API backend (default)
|
|
123
|
+
- **CLASSICUI**: Traditional server-side rendered Plone
|
|
124
|
+
|
|
125
|
+
Properties:
|
|
126
|
+
- `backendServiceName` - Name of the backend Kubernetes service
|
|
127
|
+
- `frontendServiceName` - Name of the frontend service (VOLTO only)
|
|
128
|
+
- `variant` - Deployment variant (VOLTO or CLASSICUI)
|
|
129
|
+
- `siteId` - Plone site ID in ZODB (default: 'Plone')
|
|
130
|
+
|
|
131
|
+
#### `PloneHttpcache`
|
|
132
|
+
|
|
133
|
+
Varnish HTTP caching layer using the [kube-httpcache](https://github.com/mittwald/kube-httpcache) Helm chart.
|
|
134
|
+
|
|
135
|
+
Properties:
|
|
136
|
+
- `httpcacheServiceName` - Name of the Varnish service
|
|
137
|
+
|
|
138
|
+
### Configuration Options
|
|
139
|
+
|
|
140
|
+
#### `PloneOptions`
|
|
141
|
+
|
|
142
|
+
- `version` - Version of your project
|
|
143
|
+
- `siteId` - Plone site ID (default: 'Plone')
|
|
144
|
+
- `variant` - PloneVariant.VOLTO or PloneVariant.CLASSICUI (default: VOLTO)
|
|
145
|
+
- `backend` - Backend configuration (PloneBaseOptions)
|
|
146
|
+
- `frontend` - Frontend configuration (PloneBaseOptions, required for VOLTO)
|
|
147
|
+
- `imagePullSecrets` - Image pull secrets for private registries
|
|
148
|
+
|
|
149
|
+
#### `PloneBaseOptions`
|
|
150
|
+
|
|
151
|
+
Configuration for backend or frontend:
|
|
152
|
+
|
|
153
|
+
**Container:**
|
|
154
|
+
- `image` - Container image (e.g., 'plone/plone-backend:6.0.10')
|
|
155
|
+
- `imagePullPolicy` - Pull policy (default: 'IfNotPresent')
|
|
156
|
+
- `replicas` - Number of replicas (default: 2)
|
|
157
|
+
- `environment` - Environment variables (cdk8s-plus-30.Env)
|
|
158
|
+
|
|
159
|
+
**Resources:**
|
|
160
|
+
- `requestCpu` / `limitCpu` - CPU requests/limits
|
|
161
|
+
- `requestMemory` / `limitMemory` - Memory requests/limits
|
|
162
|
+
|
|
163
|
+
**High Availability:**
|
|
164
|
+
- `minAvailable` - Min pods during updates (for PodDisruptionBudget)
|
|
165
|
+
- `maxUnavailable` - Max unavailable pods during updates
|
|
166
|
+
|
|
167
|
+
**Health Probes:**
|
|
168
|
+
- `readinessEnabled` - Enable readiness probe (default: true)
|
|
169
|
+
- `readinessInitialDelaySeconds` / `readinessTimeoutSeconds` / `readinessPeriodSeconds`
|
|
170
|
+
- `readinessSuccessThreshold` / `readinessFailureThreshold`
|
|
171
|
+
- `livenessEnabled` - Enable liveness probe (default: false, recommended true for frontend)
|
|
172
|
+
- `livenessInitialDelaySeconds` / `livenessTimeoutSeconds` / `livenessPeriodSeconds`
|
|
173
|
+
- `livenessSuccessThreshold` / `livenessFailureThreshold`
|
|
174
|
+
|
|
175
|
+
**Annotations:**
|
|
176
|
+
- `annotations` - Deployment metadata annotations
|
|
177
|
+
- `podAnnotations` - Pod template annotations (e.g., for Prometheus)
|
|
178
|
+
- `serviceAnnotations` - Service annotations (e.g., for external-dns)
|
|
179
|
+
|
|
180
|
+
#### `PloneHttpcacheOptions`
|
|
181
|
+
|
|
182
|
+
- `plone` - Plone construct to attach cache to (required)
|
|
183
|
+
- `varnishVcl` - VCL configuration as string
|
|
184
|
+
- `varnishVclFile` - Path to VCL configuration file
|
|
185
|
+
- `existingSecret` - Kubernetes secret for Varnish admin credentials
|
|
186
|
+
- `replicas` - Number of Varnish replicas (default: 2)
|
|
187
|
+
- `requestCpu` / `limitCpu` - CPU resources
|
|
188
|
+
- `requestMemory` / `limitMemory` - Memory resources
|
|
189
|
+
- `servicemonitor` - Enable Prometheus ServiceMonitor (default: false)
|
|
190
|
+
- `exporterEnabled` - Enable Prometheus exporter sidecar (default: true)
|
|
191
|
+
- `chartVersion` - kube-httpcache Helm chart version (default: latest)
|
|
174
192
|
|
|
175
193
|
|
|
176
194
|
## Development
|
|
177
195
|
|
|
178
|
-
|
|
196
|
+
This project uses [Projen](https://projen.io/) to manage project configuration. **Do not edit generated files directly.**
|
|
197
|
+
|
|
198
|
+
### Setup
|
|
199
|
+
|
|
200
|
+
Clone the repository and install dependencies:
|
|
179
201
|
|
|
180
202
|
```bash
|
|
181
203
|
nvm use lts/*
|
|
@@ -183,11 +205,42 @@ corepack enable
|
|
|
183
205
|
npx projen
|
|
184
206
|
```
|
|
185
207
|
|
|
186
|
-
|
|
208
|
+
### Common Commands
|
|
187
209
|
|
|
188
210
|
```bash
|
|
211
|
+
# Run tests
|
|
189
212
|
npx projen test
|
|
213
|
+
|
|
214
|
+
# Run tests in watch mode
|
|
215
|
+
npx projen test:watch
|
|
216
|
+
|
|
217
|
+
# Build (compile TypeScript + generate JSII bindings)
|
|
218
|
+
npx projen build
|
|
219
|
+
|
|
220
|
+
# Lint
|
|
221
|
+
npx projen eslint
|
|
222
|
+
|
|
223
|
+
# Generate API documentation
|
|
224
|
+
npx projen docgen
|
|
225
|
+
|
|
226
|
+
# Package for distribution
|
|
227
|
+
npx projen package-all
|
|
190
228
|
```
|
|
191
229
|
|
|
192
|
-
|
|
193
|
-
|
|
230
|
+
### Making Changes
|
|
231
|
+
|
|
232
|
+
1. Edit `.projenrc.ts` for project configuration changes
|
|
233
|
+
2. Run `npx projen` to regenerate project files
|
|
234
|
+
3. Make code changes in `src/`
|
|
235
|
+
4. Run tests and update snapshots if needed: `npx projen test -- -u`
|
|
236
|
+
|
|
237
|
+
## References
|
|
238
|
+
|
|
239
|
+
- [CDK8S Documentation](https://cdk8s.io/)
|
|
240
|
+
- [Kubernetes Probes Documentation](https://kubernetes.io/docs/concepts/configuration/liveness-readiness-startup-probes/)
|
|
241
|
+
- [kube-httpcache Helm Chart](https://github.com/mittwald/kube-httpcache)
|
|
242
|
+
- [Example Project](https://github.com/bluedynamics/cdk8s-plone-example)
|
|
243
|
+
|
|
244
|
+
## License
|
|
245
|
+
|
|
246
|
+
See [LICENSE](./LICENSE) file.
|
package/lib/deployment.d.ts
CHANGED
|
@@ -2,97 +2,133 @@ import * as kplus from 'cdk8s-plus-30';
|
|
|
2
2
|
import { Construct } from 'constructs';
|
|
3
3
|
import * as k8s from './imports/k8s';
|
|
4
4
|
import { PlonePDBOptions } from './pdb';
|
|
5
|
+
/**
|
|
6
|
+
* Container image configuration options.
|
|
7
|
+
*/
|
|
5
8
|
export interface PloneImageOptions {
|
|
6
9
|
/**
|
|
7
|
-
*
|
|
8
|
-
* @
|
|
10
|
+
* Container image name and tag.
|
|
11
|
+
* @example 'plone/plone-backend:6.0.10'
|
|
12
|
+
* @default 'plone/plone-backend:latest'
|
|
9
13
|
*/
|
|
10
14
|
readonly image?: string;
|
|
11
15
|
/**
|
|
12
|
-
*
|
|
13
|
-
* @default
|
|
16
|
+
* Names of Kubernetes secrets for pulling private images.
|
|
17
|
+
* @default []
|
|
14
18
|
*/
|
|
15
19
|
readonly imagePullSecrets?: string[];
|
|
16
20
|
/**
|
|
17
|
-
*
|
|
18
|
-
* @default
|
|
21
|
+
* Image pull policy (Always, IfNotPresent, Never).
|
|
22
|
+
* @default 'IfNotPresent'
|
|
19
23
|
*/
|
|
20
24
|
readonly imagePullPolicy?: string;
|
|
21
25
|
}
|
|
26
|
+
/**
|
|
27
|
+
* Configuration options for PloneDeployment.
|
|
28
|
+
*/
|
|
22
29
|
export interface PloneDeploymentOptions {
|
|
23
30
|
/**
|
|
24
|
-
*
|
|
25
|
-
* @default
|
|
31
|
+
* Container image configuration.
|
|
32
|
+
* @default 'plone/plone-backend:latest'
|
|
26
33
|
*/
|
|
27
34
|
readonly image?: PloneImageOptions;
|
|
28
35
|
/**
|
|
29
|
-
*
|
|
30
|
-
*
|
|
31
|
-
|
|
36
|
+
* Environment variables for the container.
|
|
37
|
+
* Use cdk8s-plus-30 Env to define variables and sources.
|
|
38
|
+
* @default - no additional environment variables
|
|
39
|
+
*/
|
|
32
40
|
readonly environment?: kplus.Env;
|
|
33
41
|
/**
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
42
|
+
* Number of pod replicas to run.
|
|
43
|
+
* @default 2
|
|
44
|
+
*/
|
|
37
45
|
readonly replicas?: number;
|
|
38
46
|
/**
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
47
|
+
* CPU limit for the container.
|
|
48
|
+
* @default '1000m'
|
|
49
|
+
*/
|
|
42
50
|
readonly limitCpu?: string;
|
|
43
51
|
/**
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
52
|
+
* Memory limit for the container.
|
|
53
|
+
* @default '1Gi'
|
|
54
|
+
*/
|
|
47
55
|
readonly limitMemory?: string;
|
|
48
56
|
/**
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
57
|
+
* CPU request for the container.
|
|
58
|
+
* @default '200m'
|
|
59
|
+
*/
|
|
52
60
|
readonly requestCpu?: string;
|
|
53
61
|
/**
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
62
|
+
* Memory request for the container.
|
|
63
|
+
* @default '300Mi'
|
|
64
|
+
*/
|
|
57
65
|
readonly requestMemory?: string;
|
|
58
66
|
/**
|
|
59
|
-
*
|
|
67
|
+
* Container port number to expose.
|
|
60
68
|
*/
|
|
61
69
|
readonly port: number;
|
|
62
70
|
/**
|
|
63
|
-
*
|
|
64
|
-
* @default -
|
|
71
|
+
* Additional Kubernetes labels for the deployment.
|
|
72
|
+
* @default - standard Plone labels only
|
|
65
73
|
*/
|
|
66
74
|
readonly labels?: {
|
|
67
75
|
[name: string]: string;
|
|
68
76
|
};
|
|
69
77
|
/**
|
|
70
|
-
*
|
|
71
|
-
*
|
|
78
|
+
* Annotations to add to the Deployment metadata.
|
|
79
|
+
* Common annotations include: deployment timestamps, change tracking, etc.
|
|
80
|
+
* @example { 'deployment.kubernetes.io/revision': '1' }
|
|
81
|
+
* @default - no additional annotations
|
|
82
|
+
*/
|
|
83
|
+
readonly annotations?: {
|
|
84
|
+
[name: string]: string;
|
|
85
|
+
};
|
|
86
|
+
/**
|
|
87
|
+
* Annotations to add to the Pod template metadata.
|
|
88
|
+
* Common annotations include: Prometheus scraping config, Istio config,
|
|
89
|
+
* backup policies, logging configurations, etc.
|
|
90
|
+
* @example { 'prometheus.io/scrape': 'true', 'prometheus.io/port': '8080' }
|
|
91
|
+
* @default - no additional annotations
|
|
92
|
+
*/
|
|
93
|
+
readonly podAnnotations?: {
|
|
94
|
+
[name: string]: string;
|
|
95
|
+
};
|
|
96
|
+
/**
|
|
97
|
+
* Additional container specification overrides.
|
|
98
|
+
* Advanced use only - merges with generated container spec.
|
|
99
|
+
* @default - undefined
|
|
72
100
|
*/
|
|
73
101
|
readonly ploneContainer?: k8s.Container;
|
|
74
102
|
/**
|
|
75
|
-
* Sidecar
|
|
76
|
-
* @
|
|
103
|
+
* Sidecar containers to run alongside the main container.
|
|
104
|
+
* @example [{ name: 'log-forwarder', image: 'fluentd:latest' }]
|
|
105
|
+
* @default []
|
|
77
106
|
*/
|
|
78
107
|
readonly sidecars?: k8s.Container[];
|
|
79
108
|
/**
|
|
80
|
-
*
|
|
81
|
-
* If
|
|
82
|
-
* @default -
|
|
109
|
+
* PodDisruptionBudget configuration for high availability.
|
|
110
|
+
* If provided, creates a PDB with the specified constraints.
|
|
111
|
+
* @default - no PDB created
|
|
83
112
|
*/
|
|
84
113
|
readonly pdb?: PlonePDBOptions;
|
|
85
114
|
/**
|
|
86
|
-
* Liveness
|
|
87
|
-
* @default -
|
|
115
|
+
* Liveness probe configuration for the container.
|
|
116
|
+
* @default - undefined (no liveness probe)
|
|
88
117
|
*/
|
|
89
118
|
livenessProbe?: k8s.Probe;
|
|
90
119
|
/**
|
|
91
|
-
* Readiness
|
|
92
|
-
* @default -
|
|
120
|
+
* Readiness probe configuration for the container.
|
|
121
|
+
* @default - undefined (no readiness probe)
|
|
93
122
|
*/
|
|
94
123
|
readinessProbe?: k8s.Probe;
|
|
95
124
|
}
|
|
125
|
+
/**
|
|
126
|
+
* PloneDeployment creates a Kubernetes Deployment for Plone containers.
|
|
127
|
+
*
|
|
128
|
+
* This is an internal construct used by the Plone class.
|
|
129
|
+
* It creates a Deployment with configurable replicas, resources, probes,
|
|
130
|
+
* and an optional PodDisruptionBudget.
|
|
131
|
+
*/
|
|
96
132
|
export declare class PloneDeployment extends Construct {
|
|
97
133
|
constructor(scope: Construct, id: string, options: PloneDeploymentOptions);
|
|
98
134
|
}
|
package/lib/deployment.js
CHANGED
|
@@ -7,6 +7,13 @@ const kplus = require("cdk8s-plus-30");
|
|
|
7
7
|
const constructs_1 = require("constructs");
|
|
8
8
|
const k8s = require("./imports/k8s");
|
|
9
9
|
const pdb_1 = require("./pdb");
|
|
10
|
+
/**
|
|
11
|
+
* PloneDeployment creates a Kubernetes Deployment for Plone containers.
|
|
12
|
+
*
|
|
13
|
+
* This is an internal construct used by the Plone class.
|
|
14
|
+
* It creates a Deployment with configurable replicas, resources, probes,
|
|
15
|
+
* and an optional PodDisruptionBudget.
|
|
16
|
+
*/
|
|
10
17
|
class PloneDeployment extends constructs_1.Construct {
|
|
11
18
|
constructor(scope, id, options) {
|
|
12
19
|
super(scope, id);
|
|
@@ -56,6 +63,7 @@ class PloneDeployment extends constructs_1.Construct {
|
|
|
56
63
|
const deploymentOptions = {
|
|
57
64
|
metadata: {
|
|
58
65
|
labels: deploymentLabels,
|
|
66
|
+
annotations: options.annotations,
|
|
59
67
|
},
|
|
60
68
|
spec: {
|
|
61
69
|
replicas,
|
|
@@ -63,7 +71,10 @@ class PloneDeployment extends constructs_1.Construct {
|
|
|
63
71
|
matchLabels: label,
|
|
64
72
|
},
|
|
65
73
|
template: {
|
|
66
|
-
metadata: {
|
|
74
|
+
metadata: {
|
|
75
|
+
labels: template_labels,
|
|
76
|
+
annotations: options.podAnnotations,
|
|
77
|
+
},
|
|
67
78
|
spec: {
|
|
68
79
|
imagePullSecrets: (image.imagePullSecrets ?? []).map((name) => ({ name: name })),
|
|
69
80
|
containers: [
|
|
@@ -82,4 +93,4 @@ class PloneDeployment extends constructs_1.Construct {
|
|
|
82
93
|
}
|
|
83
94
|
}
|
|
84
95
|
exports.PloneDeployment = PloneDeployment;
|
|
85
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deployment.js","sourceRoot":"","sources":["../src/deployment.ts"],"names":[],"mappings":"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","sourcesContent":["// import { log } from 'console';\nimport { Names } from 'cdk8s';\nimport * as kplus from 'cdk8s-plus-30';\nimport { Construct } from 'constructs';\nimport * as k8s from './imports/k8s';\nimport { PlonePDB, PlonePDBOptions } from './pdb';\n\nexport interface PloneImageOptions {\n  /**\n   * Specify a custom image for Plone .\n   * @default \"\"\n   */\n  readonly image?: string;\n\n  /**\n   * Specify Secret to pull image .\n   * @default \"\"\n   */\n  readonly imagePullSecrets?: string[];\n\n  /**\n   * Specify Pull Policy .\n   * @default \"\"\n   */\n  readonly imagePullPolicy?: string;\n}\n\nexport interface PloneDeploymentOptions {\n  /**\n   * Specify a custom image for Plone .\n   * @default \"plone/plone-backend:latest\"\n   */\n  readonly image?: PloneImageOptions;\n\n  /**\n   * Specify an environment for Plone .\n   * @default - none\n  */\n  readonly environment?: kplus.Env;\n\n  /**\n * Number of replicas.\n * @default 2\n */\n  readonly replicas?: number;\n\n  /**\n * CPU limit\n * @default 1\n */\n  readonly limitCpu?: string;\n\n  /**\n * memory limit\n * @default 1\n */\n  readonly limitMemory?: string;\n\n  /**\n * CPU request\n * @default 1\n */\n  readonly requestCpu?: string;\n\n  /**\n * memory request\n * @default 1\n */\n  readonly requestMemory?: string;\n\n  /**\n   * Port number.\n   */\n  readonly port: number;\n\n  /**\n   * Extra labels to associate with resources.\n   * @default - none\n   */\n  readonly labels?: { [name: string]: string };\n\n  /**\n   * Extra container spec to use for ploencontainer.\n   * @default - []\n   */\n  readonly ploneContainer?: k8s.Container;\n\n  /**\n   * Sidecar container spec to associate with resources.\n   * @default - []\n   */\n  readonly sidecars?: k8s.Container[];\n\n  /**\n   * Create a PodDisruptionBugdet for the deployment?\n   * If given\n   * @default - none\n   */\n  readonly pdb?: PlonePDBOptions;\n\n  /**\n   * Liveness Probe for the pod.\n   * @default - generated\n   */\n  livenessProbe?: k8s.Probe;\n\n  /**\n   * Readiness Probe for the pod.\n   * @default - generated\n   */\n  readinessProbe?: k8s.Probe;\n\n}\n\nexport class PloneDeployment extends Construct {\n\n  constructor(scope: Construct, id: string, options: PloneDeploymentOptions) {\n    super(scope, id);\n    const image = options.image ?? {};\n    const replicas = options.replicas ?? 2;\n    const label = { app: Names.toLabelValue(this) };\n    const optionLabels = options.labels ?? {};\n    const deploymentLabels = {\n      'app.kubernetes.io/name': optionLabels['app.kubernetes.io/name'] + '-deployment',\n      'app.kubernetes.io/component': optionLabels['app.kubernetes.io/component'] ?? '' + '-deployment',\n    };\n    const template_labels = {\n      ...optionLabels,\n      ...label,\n      'app.kubernetes.io/part-of': 'plone',\n      'app.kubernetes.io/managed-by': 'cdk8s-plone',\n    };\n    const kpEnv: kplus.Env = options?.environment ?? new kplus.Env([], {});\n    var env: k8s.EnvVar[] = [];\n    for (const name in kpEnv.variables) {\n      env.push({ name: name, value: kpEnv.variables[name].value, valueFrom: kpEnv.variables[name].valueFrom });\n    }\n    var envFrom: k8s.EnvFromSource[] = [];\n    for (const idx in kpEnv.sources) {\n      const source = kpEnv.sources[idx];\n      envFrom.push(source._toKube());\n    }\n    var ploneContainerSpec: k8s.Container = {\n      name: id + '-container', // here the namespaced name shold be used, but how?\n      image: image.image,\n      imagePullPolicy: image.imagePullPolicy,\n      env: env,\n      envFrom: envFrom,\n      resources: {\n        limits: {\n          cpu: k8s.Quantity.fromString(options.limitCpu ?? '1000m'),\n          memory: k8s.Quantity.fromString(options.limitMemory ?? '1Gi'),\n        },\n        requests: {\n          cpu: k8s.Quantity.fromString(options.requestCpu ?? '200m'),\n          memory: k8s.Quantity.fromString(options.requestMemory ?? '300Mi'),\n        },\n      },\n      livenessProbe: options.livenessProbe ?? undefined,\n      readinessProbe: options.readinessProbe ?? undefined,\n    };\n    const deploymentOptions: k8s.KubeDeploymentProps = {\n      metadata: {\n        labels: deploymentLabels,\n      },\n      spec: {\n        replicas,\n        selector: {\n          matchLabels: label,\n        },\n        template: {\n          metadata: { labels: template_labels },\n          spec: {\n            imagePullSecrets: (image.imagePullSecrets ?? []).map((name) => ({ name: name })),\n            containers: [\n              ploneContainerSpec,\n              ...options.sidecars ?? [],\n            ],\n          },\n        },\n      },\n    };\n\n    new k8s.KubeDeployment(this, 'deployment', deploymentOptions);\n\n    if (options.pdb ?? false) {\n      const pdbOptions = options.pdb ?? {};\n      new PlonePDB(this, 'pdb', label, pdbOptions);\n    }\n  }\n}\n"]}
|
|
96
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deployment.js","sourceRoot":"","sources":["../src/deployment.ts"],"names":[],"mappings":"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","sourcesContent":["// import { log } from 'console';\nimport { Names } from 'cdk8s';\nimport * as kplus from 'cdk8s-plus-30';\nimport { Construct } from 'constructs';\nimport * as k8s from './imports/k8s';\nimport { PlonePDB, PlonePDBOptions } from './pdb';\n\n/**\n * Container image configuration options.\n */\nexport interface PloneImageOptions {\n  /**\n   * Container image name and tag.\n   * @example 'plone/plone-backend:6.0.10'\n   * @default 'plone/plone-backend:latest'\n   */\n  readonly image?: string;\n\n  /**\n   * Names of Kubernetes secrets for pulling private images.\n   * @default []\n   */\n  readonly imagePullSecrets?: string[];\n\n  /**\n   * Image pull policy (Always, IfNotPresent, Never).\n   * @default 'IfNotPresent'\n   */\n  readonly imagePullPolicy?: string;\n}\n\n/**\n * Configuration options for PloneDeployment.\n */\nexport interface PloneDeploymentOptions {\n  /**\n   * Container image configuration.\n   * @default 'plone/plone-backend:latest'\n   */\n  readonly image?: PloneImageOptions;\n\n  /**\n   * Environment variables for the container.\n   * Use cdk8s-plus-30 Env to define variables and sources.\n   * @default - no additional environment variables\n   */\n  readonly environment?: kplus.Env;\n\n  /**\n   * Number of pod replicas to run.\n   * @default 2\n   */\n  readonly replicas?: number;\n\n  /**\n   * CPU limit for the container.\n   * @default '1000m'\n   */\n  readonly limitCpu?: string;\n\n  /**\n   * Memory limit for the container.\n   * @default '1Gi'\n   */\n  readonly limitMemory?: string;\n\n  /**\n   * CPU request for the container.\n   * @default '200m'\n   */\n  readonly requestCpu?: string;\n\n  /**\n   * Memory request for the container.\n   * @default '300Mi'\n   */\n  readonly requestMemory?: string;\n\n  /**\n   * Container port number to expose.\n   */\n  readonly port: number;\n\n  /**\n   * Additional Kubernetes labels for the deployment.\n   * @default - standard Plone labels only\n   */\n  readonly labels?: { [name: string]: string };\n\n  /**\n   * Annotations to add to the Deployment metadata.\n   * Common annotations include: deployment timestamps, change tracking, etc.\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 annotations include: Prometheus scraping config, Istio config,\n   * backup policies, logging configurations, 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   * Additional container specification overrides.\n   * Advanced use only - merges with generated container spec.\n   * @default - undefined\n   */\n  readonly ploneContainer?: k8s.Container;\n\n  /**\n   * Sidecar containers to run alongside the main container.\n   * @example [{ name: 'log-forwarder', image: 'fluentd:latest' }]\n   * @default []\n   */\n  readonly sidecars?: k8s.Container[];\n\n  /**\n   * PodDisruptionBudget configuration for high availability.\n   * If provided, creates a PDB with the specified constraints.\n   * @default - no PDB created\n   */\n  readonly pdb?: PlonePDBOptions;\n\n  /**\n   * Liveness probe configuration for the container.\n   * @default - undefined (no liveness probe)\n   */\n  livenessProbe?: k8s.Probe;\n\n  /**\n   * Readiness probe configuration for the container.\n   * @default - undefined (no readiness probe)\n   */\n  readinessProbe?: k8s.Probe;\n}\n\n/**\n * PloneDeployment creates a Kubernetes Deployment for Plone containers.\n *\n * This is an internal construct used by the Plone class.\n * It creates a Deployment with configurable replicas, resources, probes,\n * and an optional PodDisruptionBudget.\n */\nexport class PloneDeployment extends Construct {\n\n  constructor(scope: Construct, id: string, options: PloneDeploymentOptions) {\n    super(scope, id);\n    const image = options.image ?? {};\n    const replicas = options.replicas ?? 2;\n    const label = { app: Names.toLabelValue(this) };\n    const optionLabels = options.labels ?? {};\n    const deploymentLabels = {\n      'app.kubernetes.io/name': optionLabels['app.kubernetes.io/name'] + '-deployment',\n      'app.kubernetes.io/component': optionLabels['app.kubernetes.io/component'] ?? '' + '-deployment',\n    };\n    const template_labels = {\n      ...optionLabels,\n      ...label,\n      'app.kubernetes.io/part-of': 'plone',\n      'app.kubernetes.io/managed-by': 'cdk8s-plone',\n    };\n    const kpEnv: kplus.Env = options?.environment ?? new kplus.Env([], {});\n    var env: k8s.EnvVar[] = [];\n    for (const name in kpEnv.variables) {\n      env.push({ name: name, value: kpEnv.variables[name].value, valueFrom: kpEnv.variables[name].valueFrom });\n    }\n    var envFrom: k8s.EnvFromSource[] = [];\n    for (const idx in kpEnv.sources) {\n      const source = kpEnv.sources[idx];\n      envFrom.push(source._toKube());\n    }\n    var ploneContainerSpec: k8s.Container = {\n      name: id + '-container', // here the namespaced name shold be used, but how?\n      image: image.image,\n      imagePullPolicy: image.imagePullPolicy,\n      env: env,\n      envFrom: envFrom,\n      resources: {\n        limits: {\n          cpu: k8s.Quantity.fromString(options.limitCpu ?? '1000m'),\n          memory: k8s.Quantity.fromString(options.limitMemory ?? '1Gi'),\n        },\n        requests: {\n          cpu: k8s.Quantity.fromString(options.requestCpu ?? '200m'),\n          memory: k8s.Quantity.fromString(options.requestMemory ?? '300Mi'),\n        },\n      },\n      livenessProbe: options.livenessProbe ?? undefined,\n      readinessProbe: options.readinessProbe ?? undefined,\n    };\n    const deploymentOptions: k8s.KubeDeploymentProps = {\n      metadata: {\n        labels: deploymentLabels,\n        annotations: options.annotations,\n      },\n      spec: {\n        replicas,\n        selector: {\n          matchLabels: label,\n        },\n        template: {\n          metadata: {\n            labels: template_labels,\n            annotations: options.podAnnotations,\n          },\n          spec: {\n            imagePullSecrets: (image.imagePullSecrets ?? []).map((name) => ({ name: name })),\n            containers: [\n              ploneContainerSpec,\n              ...options.sidecars ?? [],\n            ],\n          },\n        },\n      },\n    };\n\n    new k8s.KubeDeployment(this, 'deployment', deploymentOptions);\n\n    if (options.pdb ?? false) {\n      const pdbOptions = options.pdb ?? {};\n      new PlonePDB(this, 'pdb', label, pdbOptions);\n    }\n  }\n}\n"]}
|