@dcyfr/ai-kubernetes 1.0.0

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.
Files changed (148) hide show
  1. package/.changeset/README.md +8 -0
  2. package/.changeset/config.json +11 -0
  3. package/.env.example +17 -0
  4. package/.github/workflows/ci.yml +46 -0
  5. package/.github/workflows/release.yml +82 -0
  6. package/AGENTS.md +41 -0
  7. package/CHANGELOG.md +63 -0
  8. package/CONTRIBUTING.md +30 -0
  9. package/LICENSE +21 -0
  10. package/README.md +668 -0
  11. package/SECURITY.md +718 -0
  12. package/dist/src/health/index.d.ts +6 -0
  13. package/dist/src/health/index.d.ts.map +1 -0
  14. package/dist/src/health/index.js +6 -0
  15. package/dist/src/health/index.js.map +1 -0
  16. package/dist/src/health/probes.d.ts +53 -0
  17. package/dist/src/health/probes.d.ts.map +1 -0
  18. package/dist/src/health/probes.js +107 -0
  19. package/dist/src/health/probes.js.map +1 -0
  20. package/dist/src/health/resources.d.ts +56 -0
  21. package/dist/src/health/resources.d.ts.map +1 -0
  22. package/dist/src/health/resources.js +126 -0
  23. package/dist/src/health/resources.js.map +1 -0
  24. package/dist/src/helm/chart.d.ts +47 -0
  25. package/dist/src/helm/chart.d.ts.map +1 -0
  26. package/dist/src/helm/chart.js +54 -0
  27. package/dist/src/helm/chart.js.map +1 -0
  28. package/dist/src/helm/index.d.ts +7 -0
  29. package/dist/src/helm/index.d.ts.map +1 -0
  30. package/dist/src/helm/index.js +7 -0
  31. package/dist/src/helm/index.js.map +1 -0
  32. package/dist/src/helm/template.d.ts +30 -0
  33. package/dist/src/helm/template.d.ts.map +1 -0
  34. package/dist/src/helm/template.js +117 -0
  35. package/dist/src/helm/template.js.map +1 -0
  36. package/dist/src/helm/values.d.ts +55 -0
  37. package/dist/src/helm/values.d.ts.map +1 -0
  38. package/dist/src/helm/values.js +102 -0
  39. package/dist/src/helm/values.js.map +1 -0
  40. package/dist/src/index.d.ts +18 -0
  41. package/dist/src/index.d.ts.map +1 -0
  42. package/dist/src/index.js +17 -0
  43. package/dist/src/index.js.map +1 -0
  44. package/dist/src/manifests/configmap.d.ts +55 -0
  45. package/dist/src/manifests/configmap.d.ts.map +1 -0
  46. package/dist/src/manifests/configmap.js +93 -0
  47. package/dist/src/manifests/configmap.js.map +1 -0
  48. package/dist/src/manifests/deployment.d.ts +53 -0
  49. package/dist/src/manifests/deployment.d.ts.map +1 -0
  50. package/dist/src/manifests/deployment.js +123 -0
  51. package/dist/src/manifests/deployment.js.map +1 -0
  52. package/dist/src/manifests/hpa.d.ts +32 -0
  53. package/dist/src/manifests/hpa.d.ts.map +1 -0
  54. package/dist/src/manifests/hpa.js +109 -0
  55. package/dist/src/manifests/hpa.js.map +1 -0
  56. package/dist/src/manifests/index.d.ts +10 -0
  57. package/dist/src/manifests/index.d.ts.map +1 -0
  58. package/dist/src/manifests/index.js +10 -0
  59. package/dist/src/manifests/index.js.map +1 -0
  60. package/dist/src/manifests/ingress.d.ts +36 -0
  61. package/dist/src/manifests/ingress.d.ts.map +1 -0
  62. package/dist/src/manifests/ingress.js +101 -0
  63. package/dist/src/manifests/ingress.js.map +1 -0
  64. package/dist/src/manifests/namespace.d.ts +19 -0
  65. package/dist/src/manifests/namespace.d.ts.map +1 -0
  66. package/dist/src/manifests/namespace.js +31 -0
  67. package/dist/src/manifests/namespace.js.map +1 -0
  68. package/dist/src/manifests/service.d.ts +43 -0
  69. package/dist/src/manifests/service.d.ts.map +1 -0
  70. package/dist/src/manifests/service.js +71 -0
  71. package/dist/src/manifests/service.js.map +1 -0
  72. package/dist/src/types/index.d.ts +8705 -0
  73. package/dist/src/types/index.d.ts.map +1 -0
  74. package/dist/src/types/index.js +370 -0
  75. package/dist/src/types/index.js.map +1 -0
  76. package/dist/src/utils/index.d.ts +7 -0
  77. package/dist/src/utils/index.d.ts.map +1 -0
  78. package/dist/src/utils/index.js +7 -0
  79. package/dist/src/utils/index.js.map +1 -0
  80. package/dist/src/utils/labels.d.ts +64 -0
  81. package/dist/src/utils/labels.d.ts.map +1 -0
  82. package/dist/src/utils/labels.js +98 -0
  83. package/dist/src/utils/labels.js.map +1 -0
  84. package/dist/src/utils/validation.d.ts +35 -0
  85. package/dist/src/utils/validation.d.ts.map +1 -0
  86. package/dist/src/utils/validation.js +178 -0
  87. package/dist/src/utils/validation.js.map +1 -0
  88. package/dist/src/utils/yaml.d.ts +14 -0
  89. package/dist/src/utils/yaml.d.ts.map +1 -0
  90. package/dist/src/utils/yaml.js +96 -0
  91. package/dist/src/utils/yaml.js.map +1 -0
  92. package/dist/tests/health.test.d.ts +5 -0
  93. package/dist/tests/health.test.d.ts.map +1 -0
  94. package/dist/tests/health.test.js +173 -0
  95. package/dist/tests/health.test.js.map +1 -0
  96. package/dist/tests/helm.test.d.ts +5 -0
  97. package/dist/tests/helm.test.d.ts.map +1 -0
  98. package/dist/tests/helm.test.js +189 -0
  99. package/dist/tests/helm.test.js.map +1 -0
  100. package/dist/tests/manifests.test.d.ts +5 -0
  101. package/dist/tests/manifests.test.d.ts.map +1 -0
  102. package/dist/tests/manifests.test.js +318 -0
  103. package/dist/tests/manifests.test.js.map +1 -0
  104. package/dist/tests/types.test.d.ts +5 -0
  105. package/dist/tests/types.test.d.ts.map +1 -0
  106. package/dist/tests/types.test.js +275 -0
  107. package/dist/tests/types.test.js.map +1 -0
  108. package/dist/tests/utils.test.d.ts +5 -0
  109. package/dist/tests/utils.test.d.ts.map +1 -0
  110. package/dist/tests/utils.test.js +227 -0
  111. package/dist/tests/utils.test.js.map +1 -0
  112. package/docs/API.md +1044 -0
  113. package/docs/DEPLOYMENT.md +559 -0
  114. package/docs/MONITORING.md +534 -0
  115. package/docs/SCALING.md +722 -0
  116. package/docs/SECURITY.md +718 -0
  117. package/docs/TROUBLESHOOTING.md +612 -0
  118. package/examples/helm-chart/index.ts +64 -0
  119. package/examples/microservices/index.ts +144 -0
  120. package/examples/web-app/index.ts +94 -0
  121. package/package.json +66 -0
  122. package/src/health/index.ts +29 -0
  123. package/src/health/probes.ts +140 -0
  124. package/src/health/resources.ts +152 -0
  125. package/src/helm/chart.ts +90 -0
  126. package/src/helm/index.ts +30 -0
  127. package/src/helm/template.ts +153 -0
  128. package/src/helm/values.ts +157 -0
  129. package/src/index.ts +178 -0
  130. package/src/manifests/configmap.ts +150 -0
  131. package/src/manifests/deployment.ts +163 -0
  132. package/src/manifests/hpa.ts +142 -0
  133. package/src/manifests/index.ts +57 -0
  134. package/src/manifests/ingress.ts +139 -0
  135. package/src/manifests/namespace.ts +42 -0
  136. package/src/manifests/service.ts +113 -0
  137. package/src/types/index.ts +461 -0
  138. package/src/utils/index.ts +26 -0
  139. package/src/utils/labels.ts +127 -0
  140. package/src/utils/validation.ts +225 -0
  141. package/src/utils/yaml.ts +99 -0
  142. package/tests/health.test.ts +220 -0
  143. package/tests/helm.test.ts +230 -0
  144. package/tests/manifests.test.ts +389 -0
  145. package/tests/types.test.ts +322 -0
  146. package/tests/utils.test.ts +280 -0
  147. package/tsconfig.json +19 -0
  148. package/vitest.config.ts +19 -0
@@ -0,0 +1,230 @@
1
+ /**
2
+ * Tests for helm module — chart, values, template
3
+ */
4
+
5
+ import { describe, it, expect } from 'vitest';
6
+ import {
7
+ createChart,
8
+ addDependency,
9
+ setChartVersion,
10
+ setAppVersion,
11
+ addMaintainer,
12
+ createValues,
13
+ setImage,
14
+ setValueResources,
15
+ setAutoscaling,
16
+ setIngress,
17
+ mergeValues,
18
+ renderTemplate,
19
+ deploymentTemplate,
20
+ serviceTemplate,
21
+ ingressTemplate,
22
+ } from '../src/helm/index.js';
23
+
24
+ // ─── Chart ────────────────────────────────────────────────────────
25
+
26
+ describe('createChart', () => {
27
+ it('creates a chart with defaults', () => {
28
+ const chart = createChart({ name: 'my-app' });
29
+ expect(chart.apiVersion).toBe('v2');
30
+ expect(chart.name).toBe('my-app');
31
+ expect(chart.version).toBe('0.1.0');
32
+ expect(chart.appVersion).toBe('1.0.0');
33
+ expect(chart.type).toBe('application');
34
+ });
35
+
36
+ it('creates a chart with custom values', () => {
37
+ const chart = createChart({
38
+ name: 'api',
39
+ version: '1.2.3',
40
+ appVersion: '2.0.0',
41
+ description: 'API service',
42
+ keywords: ['api', 'rest'],
43
+ });
44
+ expect(chart.version).toBe('1.2.3');
45
+ expect(chart.description).toBe('API service');
46
+ expect(chart.keywords).toEqual(['api', 'rest']);
47
+ });
48
+ });
49
+
50
+ describe('addDependency', () => {
51
+ it('adds a dependency', () => {
52
+ const chart = createChart({ name: 'app' });
53
+ const updated = addDependency(chart, 'redis', '17.x', 'https://charts.bitnami.com/bitnami');
54
+ expect(updated.dependencies).toHaveLength(1);
55
+ expect(updated.dependencies![0].name).toBe('redis');
56
+ });
57
+ });
58
+
59
+ describe('setChartVersion', () => {
60
+ it('updates version', () => {
61
+ const chart = createChart({ name: 'app' });
62
+ const updated = setChartVersion(chart, '2.0.0');
63
+ expect(updated.version).toBe('2.0.0');
64
+ });
65
+ });
66
+
67
+ describe('setAppVersion', () => {
68
+ it('updates app version', () => {
69
+ const chart = createChart({ name: 'app' });
70
+ const updated = setAppVersion(chart, '3.0.0');
71
+ expect(updated.appVersion).toBe('3.0.0');
72
+ });
73
+ });
74
+
75
+ describe('addMaintainer', () => {
76
+ it('adds a maintainer', () => {
77
+ const chart = createChart({ name: 'app' });
78
+ const updated = addMaintainer(chart, 'Alice', 'alice@example.com');
79
+ expect(updated.maintainers).toHaveLength(1);
80
+ expect(updated.maintainers![0].name).toBe('Alice');
81
+ });
82
+ });
83
+
84
+ // ─── Values ───────────────────────────────────────────────────────
85
+
86
+ describe('createValues', () => {
87
+ it('creates values with defaults', () => {
88
+ const vals = createValues({ image: { repository: 'nginx' } });
89
+ expect(vals.replicaCount).toBe(1);
90
+ expect(vals.image.repository).toBe('nginx');
91
+ expect(vals.image.tag).toBe('latest');
92
+ expect(vals.service.type).toBe('ClusterIP');
93
+ expect(vals.service.port).toBe(80);
94
+ expect(vals.ingress.enabled).toBe(false);
95
+ });
96
+
97
+ it('creates values with ingress', () => {
98
+ const vals = createValues({
99
+ image: { repository: 'api' },
100
+ ingressEnabled: true,
101
+ ingressHost: 'api.example.com',
102
+ });
103
+ expect(vals.ingress.enabled).toBe(true);
104
+ expect(vals.ingress.hosts).toHaveLength(1);
105
+ expect(vals.ingress.hosts[0].host).toBe('api.example.com');
106
+ });
107
+ });
108
+
109
+ describe('setImage', () => {
110
+ it('updates image repository and tag', () => {
111
+ const vals = createValues({ image: { repository: 'nginx' } });
112
+ const updated = setImage(vals, 'node', '20-alpine');
113
+ expect(updated.image.repository).toBe('node');
114
+ expect(updated.image.tag).toBe('20-alpine');
115
+ });
116
+ });
117
+
118
+ describe('setValueResources', () => {
119
+ it('sets resource requirements', () => {
120
+ const vals = createValues({ image: { repository: 'nginx' } });
121
+ const updated = setValueResources(vals, {
122
+ requests: { cpu: '100m', memory: '128Mi' },
123
+ limits: { cpu: '500m' },
124
+ });
125
+ expect(updated.resources?.requests?.cpu).toBe('100m');
126
+ expect(updated.resources?.limits?.cpu).toBe('500m');
127
+ });
128
+ });
129
+
130
+ describe('setAutoscaling', () => {
131
+ it('enables autoscaling', () => {
132
+ const vals = createValues({ image: { repository: 'nginx' } });
133
+ const updated = setAutoscaling(vals, true, { min: 2, max: 10, cpu: 80 });
134
+ expect(updated.autoscaling?.enabled).toBe(true);
135
+ expect(updated.autoscaling?.minReplicas).toBe(2);
136
+ expect(updated.autoscaling?.maxReplicas).toBe(10);
137
+ expect(updated.autoscaling?.targetCPUUtilizationPercentage).toBe(80);
138
+ });
139
+ });
140
+
141
+ describe('setIngress', () => {
142
+ it('enables and configures ingress', () => {
143
+ const vals = createValues({ image: { repository: 'nginx' } });
144
+ const updated = setIngress(vals, 'app.example.com', 'nginx');
145
+ expect(updated.ingress.enabled).toBe(true);
146
+ expect(updated.ingress.className).toBe('nginx');
147
+ expect(updated.ingress.hosts[0].host).toBe('app.example.com');
148
+ });
149
+ });
150
+
151
+ describe('mergeValues', () => {
152
+ it('merges partial overrides', () => {
153
+ const base = createValues({ image: { repository: 'nginx' } });
154
+ const merged = mergeValues(base, { replicaCount: 3 });
155
+ expect(merged.replicaCount).toBe(3);
156
+ expect(merged.image.repository).toBe('nginx');
157
+ });
158
+ });
159
+
160
+ // ─── Template ─────────────────────────────────────────────────────
161
+
162
+ describe('renderTemplate', () => {
163
+ it('replaces .Values patterns', () => {
164
+ const result = renderTemplate('replicas: {{ .Values.replicaCount }}', {
165
+ values: { replicaCount: 3 },
166
+ });
167
+ expect(result).toBe('replicas: 3');
168
+ });
169
+
170
+ it('replaces .Release.Name', () => {
171
+ const result = renderTemplate('name: {{ .Release.Name }}', {
172
+ values: {},
173
+ releaseName: 'prod',
174
+ });
175
+ expect(result).toBe('name: prod');
176
+ });
177
+
178
+ it('replaces .Chart.Name', () => {
179
+ const result = renderTemplate('chart: {{ .Chart.Name }}', {
180
+ values: {},
181
+ chartName: 'webapp',
182
+ });
183
+ expect(result).toBe('chart: webapp');
184
+ });
185
+
186
+ it('replaces .Release.Namespace', () => {
187
+ const result = renderTemplate('ns: {{ .Release.Namespace }}', {
188
+ values: {},
189
+ namespace: 'production',
190
+ });
191
+ expect(result).toBe('ns: production');
192
+ });
193
+
194
+ it('handles nested values', () => {
195
+ const result = renderTemplate('img: {{ .Values.image.repository }}', {
196
+ values: { image: { repository: 'nginx' } },
197
+ });
198
+ expect(result).toBe('img: nginx');
199
+ });
200
+
201
+ it('handles missing values gracefully', () => {
202
+ const result = renderTemplate('x: {{ .Values.missing }}', { values: {} });
203
+ expect(result).toBe('x: ');
204
+ });
205
+ });
206
+
207
+ describe('deploymentTemplate', () => {
208
+ it('returns a valid template string', () => {
209
+ const tpl = deploymentTemplate();
210
+ expect(tpl).toContain('apiVersion: apps/v1');
211
+ expect(tpl).toContain('{{ .Values.replicaCount }}');
212
+ expect(tpl).toContain('{{ .Release.Name }}');
213
+ });
214
+ });
215
+
216
+ describe('serviceTemplate', () => {
217
+ it('returns a valid template string', () => {
218
+ const tpl = serviceTemplate();
219
+ expect(tpl).toContain('kind: Service');
220
+ expect(tpl).toContain('{{ .Values.service.type }}');
221
+ });
222
+ });
223
+
224
+ describe('ingressTemplate', () => {
225
+ it('returns a valid template string', () => {
226
+ const tpl = ingressTemplate();
227
+ expect(tpl).toContain('networking.k8s.io/v1');
228
+ expect(tpl).toContain('{{ .Values.ingress.host }}');
229
+ });
230
+ });
@@ -0,0 +1,389 @@
1
+ /**
2
+ * Tests for manifests module — deployment, service, configmap, ingress, namespace, hpa
3
+ */
4
+
5
+ import { describe, it, expect } from 'vitest';
6
+ import {
7
+ createDeployment,
8
+ setReplicas,
9
+ addContainer,
10
+ setResources,
11
+ setStrategy,
12
+ createService,
13
+ clusterIPService,
14
+ nodePortService,
15
+ loadBalancerService,
16
+ addServicePort,
17
+ setServiceType,
18
+ createConfigMap,
19
+ setConfigData,
20
+ removeConfigData,
21
+ configMapFromObject,
22
+ createSecret,
23
+ secretFromStrings,
24
+ tlsSecret,
25
+ setSecretData,
26
+ createIngress,
27
+ addIngressRule,
28
+ addIngressTLS,
29
+ setIngressClass,
30
+ createNamespace,
31
+ appNamespace,
32
+ createHPA,
33
+ setCPUTarget,
34
+ setMemoryTarget,
35
+ setScaleRange,
36
+ } from '../src/manifests/index.js';
37
+
38
+ // ─── Deployment ───────────────────────────────────────────────────
39
+
40
+ describe('createDeployment', () => {
41
+ it('creates a basic deployment', () => {
42
+ const dep = createDeployment({ name: 'web', image: 'nginx:latest' });
43
+ expect(dep.apiVersion).toBe('apps/v1');
44
+ expect(dep.kind).toBe('Deployment');
45
+ expect(dep.metadata.name).toBe('web');
46
+ expect(dep.spec.replicas).toBe(1);
47
+ expect(dep.spec.template.spec.containers).toHaveLength(1);
48
+ expect(dep.spec.template.spec.containers[0].image).toBe('nginx:latest');
49
+ });
50
+
51
+ it('creates deployment with options', () => {
52
+ const dep = createDeployment({
53
+ name: 'api',
54
+ image: 'node:20',
55
+ replicas: 3,
56
+ port: 3000,
57
+ namespace: 'production',
58
+ labels: { version: 'v1' },
59
+ });
60
+ expect(dep.spec.replicas).toBe(3);
61
+ expect(dep.metadata.namespace).toBe('production');
62
+ expect(dep.metadata.labels?.version).toBe('v1');
63
+ expect(dep.spec.template.spec.containers[0].ports).toHaveLength(1);
64
+ expect(dep.spec.template.spec.containers[0].ports![0].containerPort).toBe(3000);
65
+ });
66
+
67
+ it('sets app label in selector', () => {
68
+ const dep = createDeployment({ name: 'web', image: 'nginx' });
69
+ expect(dep.spec.selector.matchLabels.app).toBe('web');
70
+ expect(dep.spec.template.metadata?.labels?.app).toBe('web');
71
+ });
72
+ });
73
+
74
+ describe('setReplicas', () => {
75
+ it('changes replica count', () => {
76
+ const dep = createDeployment({ name: 'web', image: 'nginx' });
77
+ const updated = setReplicas(dep, 5);
78
+ expect(updated.spec.replicas).toBe(5);
79
+ expect(dep.spec.replicas).toBe(1); // immutable
80
+ });
81
+ });
82
+
83
+ describe('addContainer', () => {
84
+ it('adds a sidecar container', () => {
85
+ const dep = createDeployment({ name: 'web', image: 'nginx' });
86
+ const updated = addContainer(dep, {
87
+ name: 'sidecar',
88
+ image: 'envoy:latest',
89
+ imagePullPolicy: 'IfNotPresent',
90
+ });
91
+ expect(updated.spec.template.spec.containers).toHaveLength(2);
92
+ expect(updated.spec.template.spec.containers[1].name).toBe('sidecar');
93
+ });
94
+ });
95
+
96
+ describe('setResources', () => {
97
+ it('sets resource limits on primary container', () => {
98
+ const dep = createDeployment({ name: 'web', image: 'nginx' });
99
+ const updated = setResources(dep, {
100
+ requests: { cpu: '100m', memory: '128Mi' },
101
+ limits: { cpu: '500m', memory: '512Mi' },
102
+ });
103
+ expect(updated.spec.template.spec.containers[0].resources?.limits?.cpu).toBe('500m');
104
+ });
105
+ });
106
+
107
+ describe('setStrategy', () => {
108
+ it('sets RollingUpdate strategy', () => {
109
+ const dep = createDeployment({ name: 'web', image: 'nginx' });
110
+ const updated = setStrategy(dep, 'RollingUpdate', {
111
+ maxSurge: 1,
112
+ maxUnavailable: 0,
113
+ });
114
+ expect(updated.spec.strategy?.type).toBe('RollingUpdate');
115
+ expect(updated.spec.strategy?.rollingUpdate?.maxSurge).toBe(1);
116
+ });
117
+
118
+ it('sets Recreate strategy', () => {
119
+ const dep = createDeployment({ name: 'web', image: 'nginx' });
120
+ const updated = setStrategy(dep, 'Recreate');
121
+ expect(updated.spec.strategy?.type).toBe('Recreate');
122
+ });
123
+ });
124
+
125
+ // ─── Service ──────────────────────────────────────────────────────
126
+
127
+ describe('createService', () => {
128
+ it('creates a basic service', () => {
129
+ const svc = createService({ name: 'web', port: 80 });
130
+ expect(svc.apiVersion).toBe('v1');
131
+ expect(svc.kind).toBe('Service');
132
+ expect(svc.spec.type).toBe('ClusterIP');
133
+ expect(svc.spec.ports).toHaveLength(1);
134
+ expect(svc.spec.ports[0].port).toBe(80);
135
+ expect(svc.spec.selector.app).toBe('web');
136
+ });
137
+ });
138
+
139
+ describe('clusterIPService', () => {
140
+ it('creates ClusterIP service', () => {
141
+ const svc = clusterIPService('api', 3000, 8080);
142
+ expect(svc.spec.type).toBe('ClusterIP');
143
+ expect(svc.spec.ports[0].targetPort).toBe(8080);
144
+ });
145
+ });
146
+
147
+ describe('nodePortService', () => {
148
+ it('creates NodePort service', () => {
149
+ const svc = nodePortService('web', 80, 30080);
150
+ expect(svc.spec.type).toBe('NodePort');
151
+ expect(svc.spec.ports[0].nodePort).toBe(30080);
152
+ });
153
+ });
154
+
155
+ describe('loadBalancerService', () => {
156
+ it('creates LoadBalancer service', () => {
157
+ const svc = loadBalancerService('lb', 443);
158
+ expect(svc.spec.type).toBe('LoadBalancer');
159
+ });
160
+ });
161
+
162
+ describe('addServicePort', () => {
163
+ it('adds a port to service', () => {
164
+ const svc = createService({ name: 'web', port: 80 });
165
+ const updated = addServicePort(svc, { port: 443, targetPort: 8443, protocol: 'TCP' });
166
+ expect(updated.spec.ports).toHaveLength(2);
167
+ });
168
+ });
169
+
170
+ describe('setServiceType', () => {
171
+ it('changes service type', () => {
172
+ const svc = createService({ name: 'web', port: 80 });
173
+ const updated = setServiceType(svc, 'LoadBalancer');
174
+ expect(updated.spec.type).toBe('LoadBalancer');
175
+ });
176
+ });
177
+
178
+ // ─── ConfigMap ────────────────────────────────────────────────────
179
+
180
+ describe('createConfigMap', () => {
181
+ it('creates a configmap', () => {
182
+ const cm = createConfigMap({ name: 'app-config', data: { key: 'value' } });
183
+ expect(cm.kind).toBe('ConfigMap');
184
+ expect(cm.data?.key).toBe('value');
185
+ });
186
+ });
187
+
188
+ describe('setConfigData', () => {
189
+ it('sets a key', () => {
190
+ const cm = createConfigMap({ name: 'cfg' });
191
+ const updated = setConfigData(cm, 'color', 'blue');
192
+ expect(updated.data?.color).toBe('blue');
193
+ });
194
+ });
195
+
196
+ describe('removeConfigData', () => {
197
+ it('removes a key', () => {
198
+ const cm = createConfigMap({ name: 'cfg', data: { a: '1', b: '2' } });
199
+ const updated = removeConfigData(cm, 'a');
200
+ expect(updated.data?.a).toBeUndefined();
201
+ expect(updated.data?.b).toBe('2');
202
+ });
203
+ });
204
+
205
+ describe('configMapFromObject', () => {
206
+ it('serializes non-string values to JSON', () => {
207
+ const cm = configMapFromObject('cfg', { port: 3000, debug: true });
208
+ expect(cm.data?.port).toBe('3000');
209
+ expect(cm.data?.debug).toBe('true');
210
+ });
211
+ });
212
+
213
+ // ─── Secret ───────────────────────────────────────────────────────
214
+
215
+ describe('createSecret', () => {
216
+ it('creates an opaque secret', () => {
217
+ const s = createSecret({ name: 'db-creds', stringData: { password: 'pass' } });
218
+ expect(s.kind).toBe('Secret');
219
+ expect(s.type).toBe('Opaque');
220
+ });
221
+ });
222
+
223
+ describe('secretFromStrings', () => {
224
+ it('creates a secret from key-value pairs', () => {
225
+ const s = secretFromStrings('creds', { user: 'admin', pass: 'secret' });
226
+ expect(s.stringData?.user).toBe('admin');
227
+ });
228
+ });
229
+
230
+ describe('tlsSecret', () => {
231
+ it('creates a TLS secret', () => {
232
+ const s = tlsSecret('tls-cert', 'cert-data', 'key-data');
233
+ expect(s.type).toBe('kubernetes.io/tls');
234
+ expect(s.stringData?.['tls.crt']).toBe('cert-data');
235
+ expect(s.stringData?.['tls.key']).toBe('key-data');
236
+ });
237
+ });
238
+
239
+ describe('setSecretData', () => {
240
+ it('adds a key to a secret', () => {
241
+ const s = createSecret({ name: 'test' });
242
+ const updated = setSecretData(s, 'token', 'abc123');
243
+ expect(updated.stringData?.token).toBe('abc123');
244
+ });
245
+ });
246
+
247
+ // ─── Ingress ──────────────────────────────────────────────────────
248
+
249
+ describe('createIngress', () => {
250
+ it('creates an ingress', () => {
251
+ const ing = createIngress({
252
+ name: 'web',
253
+ host: 'example.com',
254
+ serviceName: 'web-svc',
255
+ servicePort: 80,
256
+ });
257
+ expect(ing.apiVersion).toBe('networking.k8s.io/v1');
258
+ expect(ing.kind).toBe('Ingress');
259
+ expect(ing.spec.rules).toHaveLength(1);
260
+ expect(ing.spec.rules[0].host).toBe('example.com');
261
+ });
262
+
263
+ it('adds TLS when enabled', () => {
264
+ const ing = createIngress({
265
+ name: 'web',
266
+ host: 'example.com',
267
+ serviceName: 'web-svc',
268
+ servicePort: 80,
269
+ tls: true,
270
+ });
271
+ expect(ing.spec.tls).toHaveLength(1);
272
+ expect(ing.spec.tls![0].hosts).toContain('example.com');
273
+ });
274
+ });
275
+
276
+ describe('addIngressRule', () => {
277
+ it('adds another host rule', () => {
278
+ const ing = createIngress({
279
+ name: 'web',
280
+ host: 'a.com',
281
+ serviceName: 'svc-a',
282
+ servicePort: 80,
283
+ });
284
+ const updated = addIngressRule(ing, 'b.com', 'svc-b', 8080);
285
+ expect(updated.spec.rules).toHaveLength(2);
286
+ expect(updated.spec.rules[1].host).toBe('b.com');
287
+ });
288
+ });
289
+
290
+ describe('addIngressTLS', () => {
291
+ it('adds TLS config', () => {
292
+ const ing = createIngress({
293
+ name: 'web',
294
+ host: 'example.com',
295
+ serviceName: 'svc',
296
+ servicePort: 80,
297
+ });
298
+ const updated = addIngressTLS(ing, ['example.com'], 'tls-secret');
299
+ expect(updated.spec.tls).toHaveLength(1);
300
+ expect(updated.spec.tls![0].secretName).toBe('tls-secret');
301
+ });
302
+ });
303
+
304
+ describe('setIngressClass', () => {
305
+ it('sets ingress class name', () => {
306
+ const ing = createIngress({
307
+ name: 'web',
308
+ host: 'example.com',
309
+ serviceName: 'svc',
310
+ servicePort: 80,
311
+ });
312
+ const updated = setIngressClass(ing, 'nginx');
313
+ expect(updated.spec.ingressClassName).toBe('nginx');
314
+ });
315
+ });
316
+
317
+ // ─── Namespace ────────────────────────────────────────────────────
318
+
319
+ describe('createNamespace', () => {
320
+ it('creates a namespace', () => {
321
+ const ns = createNamespace({ name: 'prod' });
322
+ expect(ns.kind).toBe('Namespace');
323
+ expect(ns.metadata.name).toBe('prod');
324
+ });
325
+ });
326
+
327
+ describe('appNamespace', () => {
328
+ it('creates a namespace with managed-by label', () => {
329
+ const ns = appNamespace('staging', 'platform');
330
+ expect(ns.metadata.labels?.['app.kubernetes.io/managed-by']).toBe('dcyfr');
331
+ expect(ns.metadata.labels?.team).toBe('platform');
332
+ });
333
+ });
334
+
335
+ // ─── HPA ──────────────────────────────────────────────────────────
336
+
337
+ describe('createHPA', () => {
338
+ it('creates an HPA with CPU target', () => {
339
+ const hpa = createHPA({
340
+ name: 'web-hpa',
341
+ targetDeployment: 'web',
342
+ maxReplicas: 10,
343
+ cpuUtilization: 80,
344
+ });
345
+ expect(hpa.apiVersion).toBe('autoscaling/v2');
346
+ expect(hpa.kind).toBe('HorizontalPodAutoscaler');
347
+ expect(hpa.spec.scaleTargetRef.name).toBe('web');
348
+ expect(hpa.spec.maxReplicas).toBe(10);
349
+ expect(hpa.spec.metrics).toHaveLength(1);
350
+ });
351
+
352
+ it('creates an HPA with CPU and memory targets', () => {
353
+ const hpa = createHPA({
354
+ name: 'api-hpa',
355
+ targetDeployment: 'api',
356
+ maxReplicas: 20,
357
+ cpuUtilization: 70,
358
+ memoryUtilization: 80,
359
+ });
360
+ expect(hpa.spec.metrics).toHaveLength(2);
361
+ });
362
+ });
363
+
364
+ describe('setCPUTarget', () => {
365
+ it('sets CPU utilization target', () => {
366
+ const hpa = createHPA({ name: 'hpa', targetDeployment: 'web', maxReplicas: 5 });
367
+ const updated = setCPUTarget(hpa, 60);
368
+ expect(updated.spec.metrics).toHaveLength(1);
369
+ expect(updated.spec.metrics![0].resource?.target.averageUtilization).toBe(60);
370
+ });
371
+ });
372
+
373
+ describe('setMemoryTarget', () => {
374
+ it('sets memory utilization target', () => {
375
+ const hpa = createHPA({ name: 'hpa', targetDeployment: 'web', maxReplicas: 5 });
376
+ const updated = setMemoryTarget(hpa, 75);
377
+ expect(updated.spec.metrics).toHaveLength(1);
378
+ expect(updated.spec.metrics![0].resource?.name).toBe('memory');
379
+ });
380
+ });
381
+
382
+ describe('setScaleRange', () => {
383
+ it('sets min/max replicas', () => {
384
+ const hpa = createHPA({ name: 'hpa', targetDeployment: 'web', maxReplicas: 5 });
385
+ const updated = setScaleRange(hpa, 2, 15);
386
+ expect(updated.spec.minReplicas).toBe(2);
387
+ expect(updated.spec.maxReplicas).toBe(15);
388
+ });
389
+ });