@microsoft/terraform-cdk-constructs 1.3.1 → 1.5.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.
- package/.jsii +32238 -27310
- package/API.md +43714 -35942
- package/lib/azure-actiongroup/lib/action-group.js +1 -1
- package/lib/azure-activitylogalert/lib/activity-log-alert.js +1 -1
- package/lib/azure-aks/lib/aks-cluster.js +1 -1
- package/lib/azure-diagnosticsettings/lib/diagnostic-settings.js +1 -1
- package/lib/azure-dnsforwardingruleset/lib/dns-forwarding-ruleset.js +1 -1
- package/lib/azure-dnsforwardingruleset/lib/forwarding-rule.js +1 -1
- package/lib/azure-dnsforwardingruleset/lib/virtual-network-link.js +1 -1
- package/lib/azure-dnsresolver/lib/dns-resolver.js +1 -1
- package/lib/azure-dnsresolver/lib/inbound-endpoint.js +1 -1
- package/lib/azure-dnsresolver/lib/outbound-endpoint.js +1 -1
- package/lib/azure-dnszone/lib/dns-zone.js +1 -1
- package/lib/azure-metricalert/lib/metric-alert.js +1 -1
- package/lib/azure-networkinterface/lib/network-interface.js +1 -1
- package/lib/azure-networksecuritygroup/lib/network-security-group.js +1 -1
- package/lib/azure-policyassignment/lib/policy-assignment.js +1 -1
- package/lib/azure-policydefinition/lib/policy-definition.js +1 -1
- package/lib/azure-privatednszone/lib/private-dns-zone.js +1 -1
- package/lib/azure-privatednszonelink/lib/private-dns-zone-link.js +1 -1
- package/lib/azure-publicipaddress/lib/public-ip-address.js +1 -1
- package/lib/azure-resourcegroup/lib/resource-group.js +1 -1
- package/lib/azure-roleassignment/lib/role-assignment.js +1 -1
- package/lib/azure-roledefinition/lib/role-definition.js +1 -1
- package/lib/azure-storageaccount/lib/storage-account.js +1 -1
- package/lib/azure-subnet/lib/subnet.js +1 -1
- package/lib/azure-virtualmachine/lib/virtual-machine.js +1 -1
- package/lib/azure-virtualnetwork/lib/virtual-network.js +1 -1
- package/lib/azure-virtualnetworkgateway/index.d.ts +4 -0
- package/lib/azure-virtualnetworkgateway/index.js +21 -0
- package/lib/azure-virtualnetworkgateway/lib/index.d.ts +5 -0
- package/lib/azure-virtualnetworkgateway/lib/index.js +22 -0
- package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway-schemas.d.ts +32 -0
- package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway-schemas.js +298 -0
- package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway.d.ts +368 -0
- package/lib/azure-virtualnetworkgateway/lib/virtual-network-gateway.js +285 -0
- package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.integ.d.ts +12 -0
- package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.integ.js +129 -0
- package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.spec.d.ts +8 -0
- package/lib/azure-virtualnetworkgateway/test/virtual-network-gateway.spec.js +691 -0
- package/lib/azure-virtualnetworkmanager/lib/connectivity-configuration.js +1 -1
- package/lib/azure-virtualnetworkmanager/lib/index.d.ts +5 -0
- package/lib/azure-virtualnetworkmanager/lib/index.js +6 -1
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool-schemas.d.ts +24 -0
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool-schemas.js +169 -0
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool-static-cidr-schemas.d.ts +32 -0
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool-static-cidr-schemas.js +206 -0
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool-static-cidr.d.ts +170 -0
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool-static-cidr.js +214 -0
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool.d.ts +175 -0
- package/lib/azure-virtualnetworkmanager/lib/ipam-pool.js +206 -0
- package/lib/azure-virtualnetworkmanager/lib/network-group-static-member.js +1 -1
- package/lib/azure-virtualnetworkmanager/lib/network-group.js +1 -1
- package/lib/azure-virtualnetworkmanager/lib/security-admin-configuration.js +1 -1
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule-collection.js +1 -1
- package/lib/azure-virtualnetworkmanager/lib/security-admin-rule.js +1 -1
- package/lib/azure-virtualnetworkmanager/lib/utils/cidr-validator.d.ts +225 -0
- package/lib/azure-virtualnetworkmanager/lib/utils/cidr-validator.js +389 -0
- package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager.d.ts +56 -0
- package/lib/azure-virtualnetworkmanager/lib/virtual-network-manager.js +29 -2
- package/lib/azure-virtualnetworkmanager/test/cidr-validator.spec.d.ts +6 -0
- package/lib/azure-virtualnetworkmanager/test/cidr-validator.spec.js +292 -0
- package/lib/azure-virtualnetworkmanager/test/ipam-pool-static-cidr.spec.d.ts +6 -0
- package/lib/azure-virtualnetworkmanager/test/ipam-pool-static-cidr.spec.js +430 -0
- package/lib/azure-virtualnetworkmanager/test/ipam-pool.spec.d.ts +6 -0
- package/lib/azure-virtualnetworkmanager/test/ipam-pool.spec.js +372 -0
- package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.integ.d.ts +2 -1
- package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.integ.js +30 -3
- package/lib/azure-virtualnetworkmanager/test/virtual-network-manager.spec.js +105 -1
- package/lib/azure-vmss/lib/virtual-machine-scale-set.js +1 -1
- package/lib/core-azure/lib/azapi/azapi-resource.js +2 -2
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-client-config/index.js +2 -2
- package/lib/core-azure/lib/azapi/providers-azapi/data-azapi-resource/index.js +5 -5
- package/lib/core-azure/lib/azapi/providers-azapi/provider/index.js +1 -1
- package/lib/core-azure/lib/azapi/providers-azapi/resource/index.js +5 -5
- package/lib/core-azure/lib/azapi/providers-azapi/resource-action/index.js +3 -3
- package/lib/core-azure/lib/azapi/providers-azapi/update-resource/index.js +3 -3
- package/lib/core-azure/lib/azapi/schema-mapper/schema-mapper.js +1 -1
- package/lib/core-azure/lib/version-manager/api-version-manager.js +1 -1
- package/lib/core-azure/lib/version-manager/interfaces/version-interfaces.js +7 -7
- package/lib/index.d.ts +2 -0
- package/lib/index.js +4 -2
- package/lib/testing/index.js +2 -2
- package/lib/testing/lib/cleanup.js +1 -1
- package/lib/testing/lib/metadata.js +1 -1
- package/package.json +1 -1
|
@@ -0,0 +1,389 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* CIDR Validator Utility
|
|
4
|
+
*
|
|
5
|
+
* Provides comprehensive validation and parsing utilities for IPv4 CIDR notation.
|
|
6
|
+
* Used by IPAM constructs to ensure proper network address space management.
|
|
7
|
+
*/
|
|
8
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
+
exports.isValidCidr = isValidCidr;
|
|
10
|
+
exports.isPrivateRange = isPrivateRange;
|
|
11
|
+
exports.cidrsOverlap = cidrsOverlap;
|
|
12
|
+
exports.isSubnet = isSubnet;
|
|
13
|
+
exports.calculateAddressCount = calculateAddressCount;
|
|
14
|
+
exports.isValidPrefixLength = isValidPrefixLength;
|
|
15
|
+
exports.validateCidr = validateCidr;
|
|
16
|
+
exports.parseCidr = parseCidr;
|
|
17
|
+
exports.checkOverlap = checkOverlap;
|
|
18
|
+
exports.validateNoOverlaps = validateNoOverlaps;
|
|
19
|
+
exports.isContained = isContained;
|
|
20
|
+
exports.validateContainment = validateContainment;
|
|
21
|
+
exports.ipToNumber = ipToNumber;
|
|
22
|
+
exports.numberToIp = numberToIp;
|
|
23
|
+
exports.prefixToMask = prefixToMask;
|
|
24
|
+
const IPV4_REGEX = /^(\d{1,3})\.(\d{1,3})\.(\d{1,3})\.(\d{1,3})\/(\d{1,2})$/;
|
|
25
|
+
// RFC 1918 private IP ranges
|
|
26
|
+
const PRIVATE_RANGES = [
|
|
27
|
+
{ start: "10.0.0.0", end: "10.255.255.255", cidr: "10.0.0.0/8" },
|
|
28
|
+
{ start: "172.16.0.0", end: "172.31.255.255", cidr: "172.16.0.0/12" },
|
|
29
|
+
{ start: "192.168.0.0", end: "192.168.255.255", cidr: "192.168.0.0/16" },
|
|
30
|
+
];
|
|
31
|
+
/**
|
|
32
|
+
* Validates if a string is a valid CIDR notation
|
|
33
|
+
*
|
|
34
|
+
* @param cidr - CIDR string (e.g., "10.0.0.0/16")
|
|
35
|
+
* @returns boolean - true if valid CIDR format
|
|
36
|
+
*
|
|
37
|
+
* @example
|
|
38
|
+
* const valid = isValidCidr("10.0.0.0/16");
|
|
39
|
+
* console.log(valid); // true
|
|
40
|
+
*/
|
|
41
|
+
function isValidCidr(cidr) {
|
|
42
|
+
const result = validateCidr(cidr);
|
|
43
|
+
return result.valid;
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Validates if CIDR is within allowed private ranges (RFC 1918)
|
|
47
|
+
*
|
|
48
|
+
* @param cidr - CIDR string
|
|
49
|
+
* @returns boolean - true if within private IP range
|
|
50
|
+
*
|
|
51
|
+
* @example
|
|
52
|
+
* const isPrivate = isPrivateRange("10.0.0.0/16");
|
|
53
|
+
* console.log(isPrivate); // true
|
|
54
|
+
*/
|
|
55
|
+
function isPrivateRange(cidr) {
|
|
56
|
+
if (!isValidCidr(cidr)) {
|
|
57
|
+
return false;
|
|
58
|
+
}
|
|
59
|
+
const parsed = parseCidr(cidr);
|
|
60
|
+
const startNum = ipToNumber(parsed.firstIp);
|
|
61
|
+
const endNum = ipToNumber(parsed.lastIp);
|
|
62
|
+
// Check if the entire CIDR range falls within any private range
|
|
63
|
+
return PRIVATE_RANGES.some((range) => {
|
|
64
|
+
const rangeStart = ipToNumber(range.start);
|
|
65
|
+
const rangeEnd = ipToNumber(range.end);
|
|
66
|
+
return startNum >= rangeStart && endNum <= rangeEnd;
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Checks if two CIDR blocks overlap
|
|
71
|
+
*
|
|
72
|
+
* @param cidr1 - First CIDR
|
|
73
|
+
* @param cidr2 - Second CIDR
|
|
74
|
+
* @returns boolean - true if CIDRs overlap
|
|
75
|
+
*
|
|
76
|
+
* @example
|
|
77
|
+
* const overlap = cidrsOverlap("10.0.0.0/8", "10.1.0.0/16");
|
|
78
|
+
* console.log(overlap); // true
|
|
79
|
+
*/
|
|
80
|
+
function cidrsOverlap(cidr1, cidr2) {
|
|
81
|
+
return checkOverlap(cidr1, cidr2);
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* Validates if child CIDR is contained within parent CIDR
|
|
85
|
+
*
|
|
86
|
+
* @param childCidr - Child CIDR block
|
|
87
|
+
* @param parentCidr - Parent CIDR block
|
|
88
|
+
* @returns boolean - true if child is subnet of parent
|
|
89
|
+
*
|
|
90
|
+
* @example
|
|
91
|
+
* const isSubnet = isSubnet("10.1.0.0/16", "10.0.0.0/8");
|
|
92
|
+
* console.log(isSubnet); // true
|
|
93
|
+
*/
|
|
94
|
+
function isSubnet(childCidr, parentCidr) {
|
|
95
|
+
return isContained(parentCidr, childCidr);
|
|
96
|
+
}
|
|
97
|
+
/**
|
|
98
|
+
* Calculates number of IP addresses in a CIDR block
|
|
99
|
+
*
|
|
100
|
+
* @param cidr - CIDR string
|
|
101
|
+
* @returns number - Total IP addresses in the block
|
|
102
|
+
*
|
|
103
|
+
* @example
|
|
104
|
+
* const count = calculateAddressCount("10.0.0.0/24");
|
|
105
|
+
* console.log(count); // 256
|
|
106
|
+
*/
|
|
107
|
+
function calculateAddressCount(cidr) {
|
|
108
|
+
const parsed = parseCidr(cidr);
|
|
109
|
+
return parsed.totalAddresses;
|
|
110
|
+
}
|
|
111
|
+
/**
|
|
112
|
+
* Validates prefix length is within allowed range
|
|
113
|
+
*
|
|
114
|
+
* @param cidr - CIDR string
|
|
115
|
+
* @param minPrefix - Minimum allowed prefix (e.g., 8)
|
|
116
|
+
* @param maxPrefix - Maximum allowed prefix (e.g., 29)
|
|
117
|
+
* @returns boolean - true if prefix length is within range
|
|
118
|
+
*
|
|
119
|
+
* @example
|
|
120
|
+
* const valid = isValidPrefixLength("10.0.0.0/24", 8, 29);
|
|
121
|
+
* console.log(valid); // true
|
|
122
|
+
*/
|
|
123
|
+
function isValidPrefixLength(cidr, minPrefix, maxPrefix) {
|
|
124
|
+
if (!isValidCidr(cidr)) {
|
|
125
|
+
return false;
|
|
126
|
+
}
|
|
127
|
+
const parsed = parseCidr(cidr);
|
|
128
|
+
return parsed.prefix >= minPrefix && parsed.prefix <= maxPrefix;
|
|
129
|
+
}
|
|
130
|
+
/**
|
|
131
|
+
* Validate CIDR format and structure
|
|
132
|
+
*
|
|
133
|
+
* @param cidr - CIDR notation string (e.g., "10.0.0.0/8")
|
|
134
|
+
* @returns Validation result with errors and warnings
|
|
135
|
+
*
|
|
136
|
+
* @example
|
|
137
|
+
* const result = validateCidr("10.0.0.0/8");
|
|
138
|
+
* if (!result.valid) {
|
|
139
|
+
* console.error("Invalid CIDR:", result.errors);
|
|
140
|
+
* }
|
|
141
|
+
*/
|
|
142
|
+
function validateCidr(cidr) {
|
|
143
|
+
const errors = [];
|
|
144
|
+
const warnings = [];
|
|
145
|
+
if (!cidr || typeof cidr !== "string") {
|
|
146
|
+
errors.push("CIDR must be a non-empty string");
|
|
147
|
+
return { valid: false, errors, warnings };
|
|
148
|
+
}
|
|
149
|
+
const match = cidr.match(IPV4_REGEX);
|
|
150
|
+
if (!match) {
|
|
151
|
+
errors.push(`Invalid CIDR format: ${cidr}. Expected format: x.x.x.x/y`);
|
|
152
|
+
return { valid: false, errors, warnings };
|
|
153
|
+
}
|
|
154
|
+
const [, oct1, oct2, oct3, oct4, prefix] = match;
|
|
155
|
+
const octets = [oct1, oct2, oct3, oct4].map(Number);
|
|
156
|
+
const prefixLength = Number(prefix);
|
|
157
|
+
// Validate octets (0-255)
|
|
158
|
+
for (let i = 0; i < octets.length; i++) {
|
|
159
|
+
if (octets[i] < 0 || octets[i] > 255) {
|
|
160
|
+
errors.push(`Invalid octet value: ${octets[i]}. Must be between 0 and 255`);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
// Validate prefix length (0-32)
|
|
164
|
+
if (prefixLength < 0 || prefixLength > 32) {
|
|
165
|
+
errors.push(`Invalid prefix length: ${prefixLength}. Must be between 0 and 32`);
|
|
166
|
+
}
|
|
167
|
+
if (errors.length > 0) {
|
|
168
|
+
return { valid: false, errors, warnings };
|
|
169
|
+
}
|
|
170
|
+
// Verify network address alignment
|
|
171
|
+
const ipNum = ipToNumber(octets.join("."));
|
|
172
|
+
const mask = prefixToMask(prefixLength);
|
|
173
|
+
const networkNum = ipNum & mask;
|
|
174
|
+
if (ipNum !== networkNum) {
|
|
175
|
+
const correctNetwork = numberToIp(networkNum);
|
|
176
|
+
warnings.push(`IP address ${octets.join(".")} is not aligned to network boundary. ` +
|
|
177
|
+
`Network address should be ${correctNetwork}/${prefixLength}`);
|
|
178
|
+
}
|
|
179
|
+
return { valid: errors.length === 0, errors, warnings };
|
|
180
|
+
}
|
|
181
|
+
/**
|
|
182
|
+
* Parse CIDR into structured information
|
|
183
|
+
*
|
|
184
|
+
* @param cidr - CIDR notation string
|
|
185
|
+
* @returns Parsed CIDR information
|
|
186
|
+
* @throws Error if CIDR format is invalid
|
|
187
|
+
*
|
|
188
|
+
* @example
|
|
189
|
+
* const parsed = parseCidr("10.0.0.0/8");
|
|
190
|
+
* console.log(`Network: ${parsed.network}, Prefix: ${parsed.prefix}`);
|
|
191
|
+
* console.log(`Range: ${parsed.firstIp} - ${parsed.lastIp}`);
|
|
192
|
+
* console.log(`Total addresses: ${parsed.totalAddresses}`);
|
|
193
|
+
*/
|
|
194
|
+
function parseCidr(cidr) {
|
|
195
|
+
const validation = validateCidr(cidr);
|
|
196
|
+
if (!validation.valid) {
|
|
197
|
+
throw new Error(`Invalid CIDR: ${validation.errors.join(", ")}`);
|
|
198
|
+
}
|
|
199
|
+
const match = cidr.match(IPV4_REGEX);
|
|
200
|
+
const [, oct1, oct2, oct3, oct4, prefix] = match;
|
|
201
|
+
const ipAddress = `${oct1}.${oct2}.${oct3}.${oct4}`;
|
|
202
|
+
const prefixLength = Number(prefix);
|
|
203
|
+
const ipNum = ipToNumber(ipAddress);
|
|
204
|
+
const mask = prefixToMask(prefixLength);
|
|
205
|
+
const networkNum = ipNum & mask;
|
|
206
|
+
const broadcastNum = networkNum | (~mask >>> 0);
|
|
207
|
+
const totalAddresses = Math.pow(2, 32 - prefixLength);
|
|
208
|
+
const firstIpNum = networkNum;
|
|
209
|
+
const lastIpNum = broadcastNum;
|
|
210
|
+
return {
|
|
211
|
+
cidr,
|
|
212
|
+
network: numberToIp(networkNum),
|
|
213
|
+
prefix: prefixLength,
|
|
214
|
+
firstIp: numberToIp(firstIpNum),
|
|
215
|
+
lastIp: numberToIp(lastIpNum),
|
|
216
|
+
totalAddresses,
|
|
217
|
+
netmask: numberToIp(mask),
|
|
218
|
+
};
|
|
219
|
+
}
|
|
220
|
+
/**
|
|
221
|
+
* Check if two CIDRs overlap
|
|
222
|
+
*
|
|
223
|
+
* @param cidr1 - First CIDR block
|
|
224
|
+
* @param cidr2 - Second CIDR block
|
|
225
|
+
* @returns True if the CIDRs overlap
|
|
226
|
+
*
|
|
227
|
+
* @example
|
|
228
|
+
* const overlaps = checkOverlap("10.0.0.0/16", "10.0.1.0/24");
|
|
229
|
+
* console.log(overlaps); // true
|
|
230
|
+
*/
|
|
231
|
+
function checkOverlap(cidr1, cidr2) {
|
|
232
|
+
const parsed1 = parseCidr(cidr1);
|
|
233
|
+
const parsed2 = parseCidr(cidr2);
|
|
234
|
+
const start1 = ipToNumber(parsed1.firstIp);
|
|
235
|
+
const end1 = ipToNumber(parsed1.lastIp);
|
|
236
|
+
const start2 = ipToNumber(parsed2.firstIp);
|
|
237
|
+
const end2 = ipToNumber(parsed2.lastIp);
|
|
238
|
+
// Check if ranges overlap
|
|
239
|
+
return !(end1 < start2 || end2 < start1);
|
|
240
|
+
}
|
|
241
|
+
/**
|
|
242
|
+
* Validate that multiple CIDRs don't overlap
|
|
243
|
+
*
|
|
244
|
+
* @param cidrs - Array of CIDR blocks to check
|
|
245
|
+
* @returns Validation result with details of any overlaps
|
|
246
|
+
*
|
|
247
|
+
* @example
|
|
248
|
+
* const result = validateNoOverlaps([
|
|
249
|
+
* "10.0.0.0/16",
|
|
250
|
+
* "10.1.0.0/16",
|
|
251
|
+
* "10.0.1.0/24"
|
|
252
|
+
* ]);
|
|
253
|
+
* if (!result.valid) {
|
|
254
|
+
* console.error("Overlapping CIDRs:", result.errors);
|
|
255
|
+
* }
|
|
256
|
+
*/
|
|
257
|
+
function validateNoOverlaps(cidrs) {
|
|
258
|
+
const errors = [];
|
|
259
|
+
const warnings = [];
|
|
260
|
+
// First validate each CIDR individually
|
|
261
|
+
for (const cidr of cidrs) {
|
|
262
|
+
const validation = validateCidr(cidr);
|
|
263
|
+
if (!validation.valid) {
|
|
264
|
+
errors.push(`Invalid CIDR ${cidr}: ${validation.errors.join(", ")}`);
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
if (errors.length > 0) {
|
|
268
|
+
return { valid: false, errors, warnings };
|
|
269
|
+
}
|
|
270
|
+
// Check for overlaps
|
|
271
|
+
for (let i = 0; i < cidrs.length; i++) {
|
|
272
|
+
for (let j = i + 1; j < cidrs.length; j++) {
|
|
273
|
+
if (checkOverlap(cidrs[i], cidrs[j])) {
|
|
274
|
+
errors.push(`CIDRs overlap: ${cidrs[i]} and ${cidrs[j]}`);
|
|
275
|
+
}
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
return { valid: errors.length === 0, errors, warnings };
|
|
279
|
+
}
|
|
280
|
+
/**
|
|
281
|
+
* Check if a child CIDR is contained within a parent CIDR
|
|
282
|
+
*
|
|
283
|
+
* @param parentCidr - Parent CIDR block
|
|
284
|
+
* @param childCidr - Child CIDR block to check
|
|
285
|
+
* @returns True if child is fully contained in parent
|
|
286
|
+
*
|
|
287
|
+
* @example
|
|
288
|
+
* const contained = isContained("10.0.0.0/16", "10.0.1.0/24");
|
|
289
|
+
* console.log(contained); // true
|
|
290
|
+
*/
|
|
291
|
+
function isContained(parentCidr, childCidr) {
|
|
292
|
+
const parent = parseCidr(parentCidr);
|
|
293
|
+
const child = parseCidr(childCidr);
|
|
294
|
+
const parentStart = ipToNumber(parent.firstIp);
|
|
295
|
+
const parentEnd = ipToNumber(parent.lastIp);
|
|
296
|
+
const childStart = ipToNumber(child.firstIp);
|
|
297
|
+
const childEnd = ipToNumber(child.lastIp);
|
|
298
|
+
// Child must be fully within parent range
|
|
299
|
+
return childStart >= parentStart && childEnd <= parentEnd;
|
|
300
|
+
}
|
|
301
|
+
/**
|
|
302
|
+
* Validate that multiple child CIDRs are all contained within a parent CIDR
|
|
303
|
+
*
|
|
304
|
+
* @param parentCidr - Parent CIDR block
|
|
305
|
+
* @param childCidrs - Array of child CIDR blocks
|
|
306
|
+
* @returns Validation result with details of any containment violations
|
|
307
|
+
*
|
|
308
|
+
* @example
|
|
309
|
+
* const result = validateContainment("10.0.0.0/16", [
|
|
310
|
+
* "10.0.1.0/24",
|
|
311
|
+
* "10.0.2.0/24"
|
|
312
|
+
* ]);
|
|
313
|
+
* if (!result.valid) {
|
|
314
|
+
* console.error("Containment violations:", result.errors);
|
|
315
|
+
* }
|
|
316
|
+
*/
|
|
317
|
+
function validateContainment(parentCidr, childCidrs) {
|
|
318
|
+
const errors = [];
|
|
319
|
+
const warnings = [];
|
|
320
|
+
// Validate parent CIDR
|
|
321
|
+
const parentValidation = validateCidr(parentCidr);
|
|
322
|
+
if (!parentValidation.valid) {
|
|
323
|
+
errors.push(`Invalid parent CIDR ${parentCidr}: ${parentValidation.errors.join(", ")}`);
|
|
324
|
+
return { valid: false, errors, warnings };
|
|
325
|
+
}
|
|
326
|
+
// Validate each child CIDR
|
|
327
|
+
for (const childCidr of childCidrs) {
|
|
328
|
+
const childValidation = validateCidr(childCidr);
|
|
329
|
+
if (!childValidation.valid) {
|
|
330
|
+
errors.push(`Invalid child CIDR ${childCidr}: ${childValidation.errors.join(", ")}`);
|
|
331
|
+
continue;
|
|
332
|
+
}
|
|
333
|
+
// Check containment
|
|
334
|
+
if (!isContained(parentCidr, childCidr)) {
|
|
335
|
+
errors.push(`Child CIDR ${childCidr} is not contained within parent CIDR ${parentCidr}`);
|
|
336
|
+
}
|
|
337
|
+
}
|
|
338
|
+
return { valid: errors.length === 0, errors, warnings };
|
|
339
|
+
}
|
|
340
|
+
/**
|
|
341
|
+
* Convert an IP address string to a 32-bit number
|
|
342
|
+
*
|
|
343
|
+
* @param ip - IP address string (e.g., "10.0.0.1")
|
|
344
|
+
* @returns 32-bit number representation
|
|
345
|
+
*
|
|
346
|
+
* @example
|
|
347
|
+
* const num = ipToNumber("10.0.0.1");
|
|
348
|
+
* console.log(num); // 167772161
|
|
349
|
+
*/
|
|
350
|
+
function ipToNumber(ip) {
|
|
351
|
+
const octets = ip.split(".").map(Number);
|
|
352
|
+
return (((octets[0] << 24) >>> 0) +
|
|
353
|
+
((octets[1] << 16) >>> 0) +
|
|
354
|
+
((octets[2] << 8) >>> 0) +
|
|
355
|
+
octets[3]);
|
|
356
|
+
}
|
|
357
|
+
/**
|
|
358
|
+
* Convert a 32-bit number to an IP address string
|
|
359
|
+
*
|
|
360
|
+
* @param num - 32-bit number representation
|
|
361
|
+
* @returns IP address string
|
|
362
|
+
*
|
|
363
|
+
* @example
|
|
364
|
+
* const ip = numberToIp(167772161);
|
|
365
|
+
* console.log(ip); // "10.0.0.1"
|
|
366
|
+
*/
|
|
367
|
+
function numberToIp(num) {
|
|
368
|
+
return [
|
|
369
|
+
(num >>> 24) & 0xff,
|
|
370
|
+
(num >>> 16) & 0xff,
|
|
371
|
+
(num >>> 8) & 0xff,
|
|
372
|
+
num & 0xff,
|
|
373
|
+
].join(".");
|
|
374
|
+
}
|
|
375
|
+
/**
|
|
376
|
+
* Convert a prefix length to a netmask number
|
|
377
|
+
*
|
|
378
|
+
* @param prefix - Prefix length (0-32)
|
|
379
|
+
* @returns 32-bit netmask number
|
|
380
|
+
*
|
|
381
|
+
* @example
|
|
382
|
+
* const mask = prefixToMask(24);
|
|
383
|
+
* const maskIp = numberToIp(mask);
|
|
384
|
+
* console.log(maskIp); // "255.255.255.0"
|
|
385
|
+
*/
|
|
386
|
+
function prefixToMask(prefix) {
|
|
387
|
+
return prefix === 0 ? 0 : (0xffffffff << (32 - prefix)) >>> 0;
|
|
388
|
+
}
|
|
389
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cidr-validator.js","sourceRoot":"","sources":["../../../../src/azure-virtualnetworkmanager/lib/utils/cidr-validator.ts"],"names":[],"mappings":";AAAA;;;;;GAKG;;AAuDH,kCAGC;AAYD,wCAeC;AAaD,oCAEC;AAaD,4BAEC;AAYD,sDAGC;AAcD,kDAWC;AAcD,oCAqDC;AAeD,8BA6BC;AAaD,oCAWC;AAkBD,gDA0BC;AAaD,kCAWC;AAkBD,kDAmCC;AAYD,gCAQC;AAYD,gCAOC;AAaD,oCAEC;AA7aD,MAAM,UAAU,GAAG,yDAAyD,CAAC;AAE7E,6BAA6B;AAC7B,MAAM,cAAc,GAAG;IACrB,EAAE,KAAK,EAAE,UAAU,EAAE,GAAG,EAAE,gBAAgB,EAAE,IAAI,EAAE,YAAY,EAAE;IAChE,EAAE,KAAK,EAAE,YAAY,EAAE,GAAG,EAAE,gBAAgB,EAAE,IAAI,EAAE,eAAe,EAAE;IACrE,EAAE,KAAK,EAAE,aAAa,EAAE,GAAG,EAAE,iBAAiB,EAAE,IAAI,EAAE,gBAAgB,EAAE;CACzE,CAAC;AAEF;;;;;;;;;GASG;AACH,SAAgB,WAAW,CAAC,IAAY;IACtC,MAAM,MAAM,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC;IAClC,OAAO,MAAM,CAAC,KAAK,CAAC;AACtB,CAAC;AAED;;;;;;;;;GASG;AACH,SAAgB,cAAc,CAAC,IAAY;IACzC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC;QACvB,OAAO,KAAK,CAAC;IACf,CAAC;IAED,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;IAC/B,MAAM,QAAQ,GAAG,UAAU,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IAC5C,MAAM,MAAM,GAAG,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IAEzC,gEAAgE;IAChE,OAAO,cAAc,CAAC,IAAI,CAAC,CAAC,KAAK,EAAE,EAAE;QACnC,MAAM,UAAU,GAAG,UAAU,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QAC3C,MAAM,QAAQ,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACvC,OAAO,QAAQ,IAAI,UAAU,IAAI,MAAM,IAAI,QAAQ,CAAC;IACtD,CAAC,CAAC,CAAC;AACL,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAgB,YAAY,CAAC,KAAa,EAAE,KAAa;IACvD,OAAO,YAAY,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;AACpC,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAgB,QAAQ,CAAC,SAAiB,EAAE,UAAkB;IAC5D,OAAO,WAAW,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;AAC5C,CAAC;AAED;;;;;;;;;GASG;AACH,SAAgB,qBAAqB,CAAC,IAAY;IAChD,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;IAC/B,OAAO,MAAM,CAAC,cAAc,CAAC;AAC/B,CAAC;AAED;;;;;;;;;;;GAWG;AACH,SAAgB,mBAAmB,CACjC,IAAY,EACZ,SAAiB,EACjB,SAAiB;IAEjB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC;QACvB,OAAO,KAAK,CAAC;IACf,CAAC;IAED,MAAM,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;IAC/B,OAAO,MAAM,CAAC,MAAM,IAAI,SAAS,IAAI,MAAM,CAAC,MAAM,IAAI,SAAS,CAAC;AAClE,CAAC;AAED;;;;;;;;;;;GAWG;AACH,SAAgB,YAAY,CAAC,IAAY;IACvC,MAAM,MAAM,GAAa,EAAE,CAAC;IAC5B,MAAM,QAAQ,GAAa,EAAE,CAAC;IAE9B,IAAI,CAAC,IAAI,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;QACtC,MAAM,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;QAC/C,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;IAC5C,CAAC;IAED,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;IACrC,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,MAAM,CAAC,IAAI,CAAC,wBAAwB,IAAI,8BAA8B,CAAC,CAAC;QACxE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;IAC5C,CAAC;IAED,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,MAAM,CAAC,GAAG,KAAK,CAAC;IACjD,MAAM,MAAM,GAAG,CAAC,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IACpD,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;IAEpC,0BAA0B;IAC1B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACvC,IAAI,MAAM,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,GAAG,GAAG,EAAE,CAAC;YACrC,MAAM,CAAC,IAAI,CACT,wBAAwB,MAAM,CAAC,CAAC,CAAC,6BAA6B,CAC/D,CAAC;QACJ,CAAC;IACH,CAAC;IAED,gCAAgC;IAChC,IAAI,YAAY,GAAG,CAAC,IAAI,YAAY,GAAG,EAAE,EAAE,CAAC;QAC1C,MAAM,CAAC,IAAI,CACT,0BAA0B,YAAY,4BAA4B,CACnE,CAAC;IACJ,CAAC;IAED,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtB,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;IAC5C,CAAC;IAED,mCAAmC;IACnC,MAAM,KAAK,GAAG,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC;IAC3C,MAAM,IAAI,GAAG,YAAY,CAAC,YAAY,CAAC,CAAC;IACxC,MAAM,UAAU,GAAG,KAAK,GAAG,IAAI,CAAC;IAEhC,IAAI,KAAK,KAAK,UAAU,EAAE,CAAC;QACzB,MAAM,cAAc,GAAG,UAAU,CAAC,UAAU,CAAC,CAAC;QAC9C,QAAQ,CAAC,IAAI,CACX,cAAc,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,uCAAuC;YACnE,6BAA6B,cAAc,IAAI,YAAY,EAAE,CAChE,CAAC;IACJ,CAAC;IAED,OAAO,EAAE,KAAK,EAAE,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;AAC1D,CAAC;AAED;;;;;;;;;;;;GAYG;AACH,SAAgB,SAAS,CAAC,IAAY;IACpC,MAAM,UAAU,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC;IACtC,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;QACtB,MAAM,IAAI,KAAK,CAAC,iBAAiB,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACnE,CAAC;IAED,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,CAAE,CAAC;IACtC,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,MAAM,CAAC,GAAG,KAAK,CAAC;IACjD,MAAM,SAAS,GAAG,GAAG,IAAI,IAAI,IAAI,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC;IACpD,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;IAEpC,MAAM,KAAK,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC;IACpC,MAAM,IAAI,GAAG,YAAY,CAAC,YAAY,CAAC,CAAC;IACxC,MAAM,UAAU,GAAG,KAAK,GAAG,IAAI,CAAC;IAChC,MAAM,YAAY,GAAG,UAAU,GAAG,CAAC,CAAC,IAAI,KAAK,CAAC,CAAC,CAAC;IAEhD,MAAM,cAAc,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,EAAE,GAAG,YAAY,CAAC,CAAC;IACtD,MAAM,UAAU,GAAG,UAAU,CAAC;IAC9B,MAAM,SAAS,GAAG,YAAY,CAAC;IAE/B,OAAO;QACL,IAAI;QACJ,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC;QAC/B,MAAM,EAAE,YAAY;QACpB,OAAO,EAAE,UAAU,CAAC,UAAU,CAAC;QAC/B,MAAM,EAAE,UAAU,CAAC,SAAS,CAAC;QAC7B,cAAc;QACd,OAAO,EAAE,UAAU,CAAC,IAAI,CAAC;KAC1B,CAAC;AACJ,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAgB,YAAY,CAAC,KAAa,EAAE,KAAa;IACvD,MAAM,OAAO,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC;IACjC,MAAM,OAAO,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC;IAEjC,MAAM,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IAC3C,MAAM,IAAI,GAAG,UAAU,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IACxC,MAAM,MAAM,GAAG,UAAU,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IAC3C,MAAM,IAAI,GAAG,UAAU,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IAExC,0BAA0B;IAC1B,OAAO,CAAC,CAAC,IAAI,GAAG,MAAM,IAAI,IAAI,GAAG,MAAM,CAAC,CAAC;AAC3C,CAAC;AAED;;;;;;;;;;;;;;;GAeG;AACH,SAAgB,kBAAkB,CAAC,KAAe;IAChD,MAAM,MAAM,GAAa,EAAE,CAAC;IAC5B,MAAM,QAAQ,GAAa,EAAE,CAAC;IAE9B,wCAAwC;IACxC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;QACzB,MAAM,UAAU,GAAG,YAAY,CAAC,IAAI,CAAC,CAAC;QACtC,IAAI,CAAC,UAAU,CAAC,KAAK,EAAE,CAAC;YACtB,MAAM,CAAC,IAAI,CAAC,gBAAgB,IAAI,KAAK,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACvE,CAAC;IACH,CAAC;IAED,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;QACtB,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;IAC5C,CAAC;IAED,qBAAqB;IACrB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACtC,KAAK,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YAC1C,IAAI,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;gBACrC,MAAM,CAAC,IAAI,CAAC,kBAAkB,KAAK,CAAC,CAAC,CAAC,QAAQ,KAAK,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAC5D,CAAC;QACH,CAAC;IACH,CAAC;IAED,OAAO,EAAE,KAAK,EAAE,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;AAC1D,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAgB,WAAW,CAAC,UAAkB,EAAE,SAAiB;IAC/D,MAAM,MAAM,GAAG,SAAS,CAAC,UAAU,CAAC,CAAC;IACrC,MAAM,KAAK,GAAG,SAAS,CAAC,SAAS,CAAC,CAAC;IAEnC,MAAM,WAAW,GAAG,UAAU,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IAC/C,MAAM,SAAS,GAAG,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;IAC5C,MAAM,UAAU,GAAG,UAAU,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IAC7C,MAAM,QAAQ,GAAG,UAAU,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;IAE1C,0CAA0C;IAC1C,OAAO,UAAU,IAAI,WAAW,IAAI,QAAQ,IAAI,SAAS,CAAC;AAC5D,CAAC;AAED;;;;;;;;;;;;;;;GAeG;AACH,SAAgB,mBAAmB,CACjC,UAAkB,EAClB,UAAoB;IAEpB,MAAM,MAAM,GAAa,EAAE,CAAC;IAC5B,MAAM,QAAQ,GAAa,EAAE,CAAC;IAE9B,uBAAuB;IACvB,MAAM,gBAAgB,GAAG,YAAY,CAAC,UAAU,CAAC,CAAC;IAClD,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,CAAC;QAC5B,MAAM,CAAC,IAAI,CACT,uBAAuB,UAAU,KAAK,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAC3E,CAAC;QACF,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;IAC5C,CAAC;IAED,2BAA2B;IAC3B,KAAK,MAAM,SAAS,IAAI,UAAU,EAAE,CAAC;QACnC,MAAM,eAAe,GAAG,YAAY,CAAC,SAAS,CAAC,CAAC;QAChD,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;YAC3B,MAAM,CAAC,IAAI,CACT,sBAAsB,SAAS,KAAK,eAAe,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CACxE,CAAC;YACF,SAAS;QACX,CAAC;QAED,oBAAoB;QACpB,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,SAAS,CAAC,EAAE,CAAC;YACxC,MAAM,CAAC,IAAI,CACT,cAAc,SAAS,wCAAwC,UAAU,EAAE,CAC5E,CAAC;QACJ,CAAC;IACH,CAAC;IAED,OAAO,EAAE,KAAK,EAAE,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC;AAC1D,CAAC;AAED;;;;;;;;;GASG;AACH,SAAgB,UAAU,CAAC,EAAU;IACnC,MAAM,MAAM,GAAG,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IACzC,OAAO,CACL,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC;QACzB,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,CAAC;QACzB,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC;QACxB,MAAM,CAAC,CAAC,CAAC,CACV,CAAC;AACJ,CAAC;AAED;;;;;;;;;GASG;AACH,SAAgB,UAAU,CAAC,GAAW;IACpC,OAAO;QACL,CAAC,GAAG,KAAK,EAAE,CAAC,GAAG,IAAI;QACnB,CAAC,GAAG,KAAK,EAAE,CAAC,GAAG,IAAI;QACnB,CAAC,GAAG,KAAK,CAAC,CAAC,GAAG,IAAI;QAClB,GAAG,GAAG,IAAI;KACX,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;AACd,CAAC;AAED;;;;;;;;;;GAUG;AACH,SAAgB,YAAY,CAAC,MAAc;IACzC,OAAO,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,UAAU,IAAI,CAAC,EAAE,GAAG,MAAM,CAAC,CAAC,KAAK,CAAC,CAAC;AAChE,CAAC","sourcesContent":["/**\n * CIDR Validator Utility\n *\n * Provides comprehensive validation and parsing utilities for IPv4 CIDR notation.\n * Used by IPAM constructs to ensure proper network address space management.\n */\n\n/* eslint-disable no-bitwise */\n\n/**\n * Result of CIDR validation operations\n */\nexport interface CidrValidationResult {\n  /** Whether the validation passed */\n  readonly valid: boolean;\n  /** List of validation errors */\n  readonly errors: string[];\n  /** List of validation warnings */\n  readonly warnings: string[];\n}\n\n/**\n * Parsed CIDR information\n */\nexport interface ParsedCidr {\n  /** Original CIDR notation (e.g., \"10.0.0.0/8\") */\n  readonly cidr: string;\n  /** Network address (e.g., \"10.0.0.0\") */\n  readonly network: string;\n  /** Prefix length (e.g., 8) */\n  readonly prefix: number;\n  /** First usable IP address */\n  readonly firstIp: string;\n  /** Last usable IP address */\n  readonly lastIp: string;\n  /** Total number of addresses in the range */\n  readonly totalAddresses: number;\n  /** Network mask (e.g., \"255.0.0.0\") */\n  readonly netmask: string;\n}\n\nconst IPV4_REGEX = /^(\\d{1,3})\\.(\\d{1,3})\\.(\\d{1,3})\\.(\\d{1,3})\\/(\\d{1,2})$/;\n\n// RFC 1918 private IP ranges\nconst PRIVATE_RANGES = [\n  { start: \"10.0.0.0\", end: \"10.255.255.255\", cidr: \"10.0.0.0/8\" },\n  { start: \"172.16.0.0\", end: \"172.31.255.255\", cidr: \"172.16.0.0/12\" },\n  { start: \"192.168.0.0\", end: \"192.168.255.255\", cidr: \"192.168.0.0/16\" },\n];\n\n/**\n * Validates if a string is a valid CIDR notation\n *\n * @param cidr - CIDR string (e.g., \"10.0.0.0/16\")\n * @returns boolean - true if valid CIDR format\n *\n * @example\n * const valid = isValidCidr(\"10.0.0.0/16\");\n * console.log(valid); // true\n */\nexport function isValidCidr(cidr: string): boolean {\n  const result = validateCidr(cidr);\n  return result.valid;\n}\n\n/**\n * Validates if CIDR is within allowed private ranges (RFC 1918)\n *\n * @param cidr - CIDR string\n * @returns boolean - true if within private IP range\n *\n * @example\n * const isPrivate = isPrivateRange(\"10.0.0.0/16\");\n * console.log(isPrivate); // true\n */\nexport function isPrivateRange(cidr: string): boolean {\n  if (!isValidCidr(cidr)) {\n    return false;\n  }\n\n  const parsed = parseCidr(cidr);\n  const startNum = ipToNumber(parsed.firstIp);\n  const endNum = ipToNumber(parsed.lastIp);\n\n  // Check if the entire CIDR range falls within any private range\n  return PRIVATE_RANGES.some((range) => {\n    const rangeStart = ipToNumber(range.start);\n    const rangeEnd = ipToNumber(range.end);\n    return startNum >= rangeStart && endNum <= rangeEnd;\n  });\n}\n\n/**\n * Checks if two CIDR blocks overlap\n *\n * @param cidr1 - First CIDR\n * @param cidr2 - Second CIDR\n * @returns boolean - true if CIDRs overlap\n *\n * @example\n * const overlap = cidrsOverlap(\"10.0.0.0/8\", \"10.1.0.0/16\");\n * console.log(overlap); // true\n */\nexport function cidrsOverlap(cidr1: string, cidr2: string): boolean {\n  return checkOverlap(cidr1, cidr2);\n}\n\n/**\n * Validates if child CIDR is contained within parent CIDR\n *\n * @param childCidr - Child CIDR block\n * @param parentCidr - Parent CIDR block\n * @returns boolean - true if child is subnet of parent\n *\n * @example\n * const isSubnet = isSubnet(\"10.1.0.0/16\", \"10.0.0.0/8\");\n * console.log(isSubnet); // true\n */\nexport function isSubnet(childCidr: string, parentCidr: string): boolean {\n  return isContained(parentCidr, childCidr);\n}\n\n/**\n * Calculates number of IP addresses in a CIDR block\n *\n * @param cidr - CIDR string\n * @returns number - Total IP addresses in the block\n *\n * @example\n * const count = calculateAddressCount(\"10.0.0.0/24\");\n * console.log(count); // 256\n */\nexport function calculateAddressCount(cidr: string): number {\n  const parsed = parseCidr(cidr);\n  return parsed.totalAddresses;\n}\n\n/**\n * Validates prefix length is within allowed range\n *\n * @param cidr - CIDR string\n * @param minPrefix - Minimum allowed prefix (e.g., 8)\n * @param maxPrefix - Maximum allowed prefix (e.g., 29)\n * @returns boolean - true if prefix length is within range\n *\n * @example\n * const valid = isValidPrefixLength(\"10.0.0.0/24\", 8, 29);\n * console.log(valid); // true\n */\nexport function isValidPrefixLength(\n  cidr: string,\n  minPrefix: number,\n  maxPrefix: number,\n): boolean {\n  if (!isValidCidr(cidr)) {\n    return false;\n  }\n\n  const parsed = parseCidr(cidr);\n  return parsed.prefix >= minPrefix && parsed.prefix <= maxPrefix;\n}\n\n/**\n * Validate CIDR format and structure\n *\n * @param cidr - CIDR notation string (e.g., \"10.0.0.0/8\")\n * @returns Validation result with errors and warnings\n *\n * @example\n * const result = validateCidr(\"10.0.0.0/8\");\n * if (!result.valid) {\n *   console.error(\"Invalid CIDR:\", result.errors);\n * }\n */\nexport function validateCidr(cidr: string): CidrValidationResult {\n  const errors: string[] = [];\n  const warnings: string[] = [];\n\n  if (!cidr || typeof cidr !== \"string\") {\n    errors.push(\"CIDR must be a non-empty string\");\n    return { valid: false, errors, warnings };\n  }\n\n  const match = cidr.match(IPV4_REGEX);\n  if (!match) {\n    errors.push(`Invalid CIDR format: ${cidr}. Expected format: x.x.x.x/y`);\n    return { valid: false, errors, warnings };\n  }\n\n  const [, oct1, oct2, oct3, oct4, prefix] = match;\n  const octets = [oct1, oct2, oct3, oct4].map(Number);\n  const prefixLength = Number(prefix);\n\n  // Validate octets (0-255)\n  for (let i = 0; i < octets.length; i++) {\n    if (octets[i] < 0 || octets[i] > 255) {\n      errors.push(\n        `Invalid octet value: ${octets[i]}. Must be between 0 and 255`,\n      );\n    }\n  }\n\n  // Validate prefix length (0-32)\n  if (prefixLength < 0 || prefixLength > 32) {\n    errors.push(\n      `Invalid prefix length: ${prefixLength}. Must be between 0 and 32`,\n    );\n  }\n\n  if (errors.length > 0) {\n    return { valid: false, errors, warnings };\n  }\n\n  // Verify network address alignment\n  const ipNum = ipToNumber(octets.join(\".\"));\n  const mask = prefixToMask(prefixLength);\n  const networkNum = ipNum & mask;\n\n  if (ipNum !== networkNum) {\n    const correctNetwork = numberToIp(networkNum);\n    warnings.push(\n      `IP address ${octets.join(\".\")} is not aligned to network boundary. ` +\n        `Network address should be ${correctNetwork}/${prefixLength}`,\n    );\n  }\n\n  return { valid: errors.length === 0, errors, warnings };\n}\n\n/**\n * Parse CIDR into structured information\n *\n * @param cidr - CIDR notation string\n * @returns Parsed CIDR information\n * @throws Error if CIDR format is invalid\n *\n * @example\n * const parsed = parseCidr(\"10.0.0.0/8\");\n * console.log(`Network: ${parsed.network}, Prefix: ${parsed.prefix}`);\n * console.log(`Range: ${parsed.firstIp} - ${parsed.lastIp}`);\n * console.log(`Total addresses: ${parsed.totalAddresses}`);\n */\nexport function parseCidr(cidr: string): ParsedCidr {\n  const validation = validateCidr(cidr);\n  if (!validation.valid) {\n    throw new Error(`Invalid CIDR: ${validation.errors.join(\", \")}`);\n  }\n\n  const match = cidr.match(IPV4_REGEX)!;\n  const [, oct1, oct2, oct3, oct4, prefix] = match;\n  const ipAddress = `${oct1}.${oct2}.${oct3}.${oct4}`;\n  const prefixLength = Number(prefix);\n\n  const ipNum = ipToNumber(ipAddress);\n  const mask = prefixToMask(prefixLength);\n  const networkNum = ipNum & mask;\n  const broadcastNum = networkNum | (~mask >>> 0);\n\n  const totalAddresses = Math.pow(2, 32 - prefixLength);\n  const firstIpNum = networkNum;\n  const lastIpNum = broadcastNum;\n\n  return {\n    cidr,\n    network: numberToIp(networkNum),\n    prefix: prefixLength,\n    firstIp: numberToIp(firstIpNum),\n    lastIp: numberToIp(lastIpNum),\n    totalAddresses,\n    netmask: numberToIp(mask),\n  };\n}\n\n/**\n * Check if two CIDRs overlap\n *\n * @param cidr1 - First CIDR block\n * @param cidr2 - Second CIDR block\n * @returns True if the CIDRs overlap\n *\n * @example\n * const overlaps = checkOverlap(\"10.0.0.0/16\", \"10.0.1.0/24\");\n * console.log(overlaps); // true\n */\nexport function checkOverlap(cidr1: string, cidr2: string): boolean {\n  const parsed1 = parseCidr(cidr1);\n  const parsed2 = parseCidr(cidr2);\n\n  const start1 = ipToNumber(parsed1.firstIp);\n  const end1 = ipToNumber(parsed1.lastIp);\n  const start2 = ipToNumber(parsed2.firstIp);\n  const end2 = ipToNumber(parsed2.lastIp);\n\n  // Check if ranges overlap\n  return !(end1 < start2 || end2 < start1);\n}\n\n/**\n * Validate that multiple CIDRs don't overlap\n *\n * @param cidrs - Array of CIDR blocks to check\n * @returns Validation result with details of any overlaps\n *\n * @example\n * const result = validateNoOverlaps([\n *   \"10.0.0.0/16\",\n *   \"10.1.0.0/16\",\n *   \"10.0.1.0/24\"\n * ]);\n * if (!result.valid) {\n *   console.error(\"Overlapping CIDRs:\", result.errors);\n * }\n */\nexport function validateNoOverlaps(cidrs: string[]): CidrValidationResult {\n  const errors: string[] = [];\n  const warnings: string[] = [];\n\n  // First validate each CIDR individually\n  for (const cidr of cidrs) {\n    const validation = validateCidr(cidr);\n    if (!validation.valid) {\n      errors.push(`Invalid CIDR ${cidr}: ${validation.errors.join(\", \")}`);\n    }\n  }\n\n  if (errors.length > 0) {\n    return { valid: false, errors, warnings };\n  }\n\n  // Check for overlaps\n  for (let i = 0; i < cidrs.length; i++) {\n    for (let j = i + 1; j < cidrs.length; j++) {\n      if (checkOverlap(cidrs[i], cidrs[j])) {\n        errors.push(`CIDRs overlap: ${cidrs[i]} and ${cidrs[j]}`);\n      }\n    }\n  }\n\n  return { valid: errors.length === 0, errors, warnings };\n}\n\n/**\n * Check if a child CIDR is contained within a parent CIDR\n *\n * @param parentCidr - Parent CIDR block\n * @param childCidr - Child CIDR block to check\n * @returns True if child is fully contained in parent\n *\n * @example\n * const contained = isContained(\"10.0.0.0/16\", \"10.0.1.0/24\");\n * console.log(contained); // true\n */\nexport function isContained(parentCidr: string, childCidr: string): boolean {\n  const parent = parseCidr(parentCidr);\n  const child = parseCidr(childCidr);\n\n  const parentStart = ipToNumber(parent.firstIp);\n  const parentEnd = ipToNumber(parent.lastIp);\n  const childStart = ipToNumber(child.firstIp);\n  const childEnd = ipToNumber(child.lastIp);\n\n  // Child must be fully within parent range\n  return childStart >= parentStart && childEnd <= parentEnd;\n}\n\n/**\n * Validate that multiple child CIDRs are all contained within a parent CIDR\n *\n * @param parentCidr - Parent CIDR block\n * @param childCidrs - Array of child CIDR blocks\n * @returns Validation result with details of any containment violations\n *\n * @example\n * const result = validateContainment(\"10.0.0.0/16\", [\n *   \"10.0.1.0/24\",\n *   \"10.0.2.0/24\"\n * ]);\n * if (!result.valid) {\n *   console.error(\"Containment violations:\", result.errors);\n * }\n */\nexport function validateContainment(\n  parentCidr: string,\n  childCidrs: string[],\n): CidrValidationResult {\n  const errors: string[] = [];\n  const warnings: string[] = [];\n\n  // Validate parent CIDR\n  const parentValidation = validateCidr(parentCidr);\n  if (!parentValidation.valid) {\n    errors.push(\n      `Invalid parent CIDR ${parentCidr}: ${parentValidation.errors.join(\", \")}`,\n    );\n    return { valid: false, errors, warnings };\n  }\n\n  // Validate each child CIDR\n  for (const childCidr of childCidrs) {\n    const childValidation = validateCidr(childCidr);\n    if (!childValidation.valid) {\n      errors.push(\n        `Invalid child CIDR ${childCidr}: ${childValidation.errors.join(\", \")}`,\n      );\n      continue;\n    }\n\n    // Check containment\n    if (!isContained(parentCidr, childCidr)) {\n      errors.push(\n        `Child CIDR ${childCidr} is not contained within parent CIDR ${parentCidr}`,\n      );\n    }\n  }\n\n  return { valid: errors.length === 0, errors, warnings };\n}\n\n/**\n * Convert an IP address string to a 32-bit number\n *\n * @param ip - IP address string (e.g., \"10.0.0.1\")\n * @returns 32-bit number representation\n *\n * @example\n * const num = ipToNumber(\"10.0.0.1\");\n * console.log(num); // 167772161\n */\nexport function ipToNumber(ip: string): number {\n  const octets = ip.split(\".\").map(Number);\n  return (\n    ((octets[0] << 24) >>> 0) +\n    ((octets[1] << 16) >>> 0) +\n    ((octets[2] << 8) >>> 0) +\n    octets[3]\n  );\n}\n\n/**\n * Convert a 32-bit number to an IP address string\n *\n * @param num - 32-bit number representation\n * @returns IP address string\n *\n * @example\n * const ip = numberToIp(167772161);\n * console.log(ip); // \"10.0.0.1\"\n */\nexport function numberToIp(num: number): string {\n  return [\n    (num >>> 24) & 0xff,\n    (num >>> 16) & 0xff,\n    (num >>> 8) & 0xff,\n    num & 0xff,\n  ].join(\".\");\n}\n\n/**\n * Convert a prefix length to a netmask number\n *\n * @param prefix - Prefix length (0-32)\n * @returns 32-bit netmask number\n *\n * @example\n * const mask = prefixToMask(24);\n * const maskIp = numberToIp(mask);\n * console.log(maskIp); // \"255.255.255.0\"\n */\nexport function prefixToMask(prefix: number): number {\n  return prefix === 0 ? 0 : (0xffffffff << (32 - prefix)) >>> 0;\n}\n"]}
|
|
@@ -21,6 +21,7 @@ import * as cdktf from "cdktf";
|
|
|
21
21
|
import { Construct } from "constructs";
|
|
22
22
|
import { ConnectivityConfiguration } from "./connectivity-configuration";
|
|
23
23
|
import { ConnectivityGroupItem, Hub } from "./connectivity-configuration-schemas";
|
|
24
|
+
import { IpamPool } from "./ipam-pool";
|
|
24
25
|
import { NetworkGroup } from "./network-group";
|
|
25
26
|
import { SecurityAdminConfiguration } from "./security-admin-configuration";
|
|
26
27
|
import { AzapiResource, AzapiResourceProps } from "../../core-azure/lib/azapi/azapi-resource";
|
|
@@ -165,6 +166,40 @@ export interface AddSecurityAdminConfigurationProps extends AzapiResourceProps {
|
|
|
165
166
|
*/
|
|
166
167
|
readonly ignoreChanges?: string[];
|
|
167
168
|
}
|
|
169
|
+
/**
|
|
170
|
+
* Properties for adding an IpamPool via the convenience method
|
|
171
|
+
* This interface excludes networkManagerId as it's automatically set
|
|
172
|
+
*/
|
|
173
|
+
export interface AddIpamPoolProps extends AzapiResourceProps {
|
|
174
|
+
/**
|
|
175
|
+
* IP address prefixes for the pool
|
|
176
|
+
* Must be valid CIDR notation (e.g., "10.0.0.0/8")
|
|
177
|
+
* Multiple prefixes must not overlap
|
|
178
|
+
* @example ["10.0.0.0/8", "172.16.0.0/12"]
|
|
179
|
+
*/
|
|
180
|
+
readonly addressPrefixes: string[];
|
|
181
|
+
/**
|
|
182
|
+
* Optional description of the IPAM pool
|
|
183
|
+
* @example "Production IP address pool for East US region"
|
|
184
|
+
*/
|
|
185
|
+
readonly description?: string;
|
|
186
|
+
/**
|
|
187
|
+
* Optional friendly display name
|
|
188
|
+
* @example "East US Production Pool"
|
|
189
|
+
*/
|
|
190
|
+
readonly displayName?: string;
|
|
191
|
+
/**
|
|
192
|
+
* Name of parent pool for hierarchical pools
|
|
193
|
+
* Leave empty/undefined for root pools
|
|
194
|
+
* @example "root-pool"
|
|
195
|
+
*/
|
|
196
|
+
readonly parentPoolName?: string;
|
|
197
|
+
/**
|
|
198
|
+
* The lifecycle rules to ignore changes
|
|
199
|
+
* @example ["tags"]
|
|
200
|
+
*/
|
|
201
|
+
readonly ignoreChanges?: string[];
|
|
202
|
+
}
|
|
168
203
|
/**
|
|
169
204
|
* Properties for Virtual Network Manager body
|
|
170
205
|
*/
|
|
@@ -330,6 +365,27 @@ export declare class VirtualNetworkManager extends AzapiResource {
|
|
|
330
365
|
* });
|
|
331
366
|
*/
|
|
332
367
|
addSecurityAdminConfiguration(id: string, props: AddSecurityAdminConfigurationProps): SecurityAdminConfiguration;
|
|
368
|
+
/**
|
|
369
|
+
* Convenience method to create an IpamPool
|
|
370
|
+
*
|
|
371
|
+
* This is a helper method that creates an IpamPool with the networkManagerId
|
|
372
|
+
* automatically set to this Network Manager's ID. You can also create IpamPools
|
|
373
|
+
* directly using: new IpamPool(scope, id, { networkManagerId: vnm.id, ...props })
|
|
374
|
+
*
|
|
375
|
+
* @param id - The unique identifier for the IPAM pool construct
|
|
376
|
+
* @param props - IpamPool properties (networkManagerId will be set automatically)
|
|
377
|
+
* @returns The created IpamPool instance
|
|
378
|
+
*
|
|
379
|
+
* @example
|
|
380
|
+
* const ipamPool = networkManager.addIpamPool("prod-pool", {
|
|
381
|
+
* name: "production-pool",
|
|
382
|
+
* location: "eastus",
|
|
383
|
+
* addressPrefixes: ["10.0.0.0/8"],
|
|
384
|
+
* description: "Production IP address pool",
|
|
385
|
+
* displayName: "Production Pool"
|
|
386
|
+
* });
|
|
387
|
+
*/
|
|
388
|
+
addIpamPool(id: string, props: AddIpamPoolProps): IpamPool;
|
|
333
389
|
/**
|
|
334
390
|
* Applies ignore changes lifecycle rules if specified in props
|
|
335
391
|
*/
|