-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathGet-IpRange.ps1
114 lines (106 loc) · 4.91 KB
/
Get-IpRange.ps1
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
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
function Get-IpRange {
<#
.SYNOPSIS
Given a subnet in CIDR format, get all of the valid IP addresses in that range.
.DESCRIPTION
Given a subnet in CIDR format, get all of the valid IP addresses in that range.
.PARAMETER Subnets
The subnet written in CIDR format 'a.b.c.d/#' and an example would be '192.168.1.24/27'. Can be a single value, an
array of values, or values can be taken from the pipeline.
.EXAMPLE
Get-IpRange -Subnets '192.168.1.24/30'
192.168.1.25
192.168.1.26
.EXAMPLE
(Get-IpRange -Subnets '10.100.10.0/24').count
254
.EXAMPLE
'192.168.1.128/30' | Get-IpRange
192.168.1.129
192.168.1.130
.NOTES
Inspired by https://gallery.technet.microsoft.com/PowerShell-Subnet-db45ec74
* Added comment help
#>
[CmdletBinding(ConfirmImpact = 'None')]
Param(
[Parameter(Mandatory, HelpMessage = 'Please enter a subnet in the form a.b.c.d/#', ValueFromPipeline, Position = 0)]
[string[]] $Subnets
)
begin {
Write-Verbose -Message "Starting [$($MyInvocation.Mycommand)]"
}
process {
foreach ($subnet in $subnets) {
if ($subnet -match '^\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}/\d{1,2}$') {
#Split IP and subnet
$IP = ($Subnet -split '\/')[0]
[int] $SubnetBits = ($Subnet -split '\/')[1]
if ($SubnetBits -lt 7 -or $SubnetBits -gt 30) {
Write-Error -Message 'The number following the / must be between 7 and 30'
break
}
#Convert IP into binary
#Split IP into different octects and for each one, figure out the binary with leading zeros and add to the total
$Octets = $IP -split '\.'
#$IPInBinary = @()
$IPInBinary = [System.Collections.ArrayList]::new()
foreach ($Octet in $Octets) {
#convert to binary
$OctetInBinary = [convert]::ToString($Octet, 2)
#get length of binary string add leading zeros to make octet
$OctetInBinary = ('0' * (8 - ($OctetInBinary).Length) + $OctetInBinary)
$IPInBinary = $IPInBinary + $OctetInBinary
}
$IPInBinary = $IPInBinary -join ''
#Get network ID by subtracting subnet mask
$HostBits = 32 - $SubnetBits
$NetworkIDInBinary = $IPInBinary.Substring(0, $SubnetBits)
#Get host ID and get the first host ID by converting all 1s into 0s
$HostIDInBinary = $IPInBinary.Substring($SubnetBits, $HostBits)
$HostIDInBinary = $HostIDInBinary -replace '1', '0'
#Work out all the host IDs in that subnet by cycling through $i from 1 up to max $HostIDInBinary (i.e. 1s stringed up to $HostBits)
#Work out max $HostIDInBinary
$imax = [convert]::ToInt32(('1' * $HostBits), 2) - 1
$IPs = @()
$IPs = [System.Collections.ArrayList]::new()
#Next ID is first network ID converted to decimal plus $i then converted to binary
For ($i = 1 ; $i -le $imax ; $i++) {
#Convert to decimal and add $i
$NextHostIDInDecimal = ([convert]::ToInt32($HostIDInBinary, 2) + $i)
#Convert back to binary
$NextHostIDInBinary = [convert]::ToString($NextHostIDInDecimal, 2)
#Add leading zeros
#Number of zeros to add
$NoOfZerosToAdd = $HostIDInBinary.Length - $NextHostIDInBinary.Length
$NextHostIDInBinary = ('0' * $NoOfZerosToAdd) + $NextHostIDInBinary
#Work out next IP
#Add networkID to hostID
$NextIPInBinary = $NetworkIDInBinary + $NextHostIDInBinary
#Split into octets and separate by . then join
$IP = @()
For ($x = 1 ; $x -le 4 ; $x++) {
#Work out start character position
$StartCharNumber = ($x - 1) * 8
#Get octet in binary
$IPOctetInBinary = $NextIPInBinary.Substring($StartCharNumber, 8)
#Convert octet into decimal
$IPOctetInDecimal = [convert]::ToInt32($IPOctetInBinary, 2)
#Add octet to IP
$IP += $IPOctetInDecimal
}
#Separate by .
$IP = $IP -join '.'
# $IPs += $IP
$IPs.Add($IP) | Out-Null
}
Write-Output -InputObject $IPs
} else {
Write-Error -Message "Subnet [$subnet] is not in a valid format"
}
}
}
end {
Write-Verbose -Message "Ending [$($MyInvocation.Mycommand)]"
}
}