STtools.psm1

function Get-STGroupsFromCSV {
    <#
    .SYNOPSIS
    Gets/outputs group names and associated/unassociated users.
    Good for adding/removing year level or class groups for staff/students.
     
    .DESCRIPTION
    Gets user and year group information from a CSV file that has at least two columns; one with usernames, the other with group names.
    The command will output to the pipeline:
    A group name made up of the year level and your prefix/postfix/both
    A list of Active Directory User objects (like you would get from Get-ADUser) who should be in the group based on their username
    A list of Active Directory User objects (like you would get from Get-ADUser) who should NOT be in the group (all OTHER users from AD)
     
    .PARAMETER csvfile
    The source CSV file that contains a field for usernames and a field for groups
     
    .PARAMETER username_header
    The header of the field that contains usernames (Defaults to "Student Code")
     
    .PARAMETER group_header
    The header of the field that contains group (Defaults to "Roll Class Code")
     
    .PARAMETER prefix
    A Prefix to prepend to the year level to generate the desired group name (Eg. "Year" to generate "Year7"). Can also be used with postfix.
     
    .PARAMETER postfix
    A Postfix to append to the year level to generate the desired group name (Eg. "Students" to generate "7Students"). Can also be used with prefix.
 
    .PARAMETER onlynumbers
    Only the numbers are kept in the group field. Useful when you're trying to create year groups from fields that have letters (Eg. "7B" -> "7")
 
    .PARAMETER stripzeros
    Strips leading zeros on group year (Eg. "07" -> "7")
     
    .EXAMPLE
    Get-STGroupsFromCSV -csvfile \\path\to\csv\file.csv -prefix "Year" |ForEach-Object {
        Remove-ADGroupMember -Identity $_.Identity -Members $_.NonMembers -Confirm:$false
        Add-ADGroupMember -Identity $_.Identity -Members $_.Members -Confirm:$false
    }
 
    Get year groups and associated users from file.csv, for each "YearX" group, remove the users who aren't in the CSV file, add users who are in the CSV file
 
    .EXAMPLE
    Get-STGroupsFromCSV -csvfile \\path\to\csv\file.csv -postfix "Teachers" -username_header "Teacher Code" |ForEach-Object {
        Remove-ADGroupMember -Identity $_.Identity -Members $_.NonMembers -Confirm:$false
        Add-ADGroupMember -Identity $_.Identity -Members $_.Members -Confirm:$false
    }
 
 
    Get year groups and associated users from file.csv, for each "XTeachers" group, remove the users who aren't in the CSV file, add users who are in the CSV file
 
    .EXAMPLE
    Get-STGroupsFromCSV -csvfile \\CASES\share\ST_XXXX.csv
    #>

    Param(
        [Parameter(Mandatory=$true)]
        [string]
        $csvfile,
        [Parameter(Mandatory=$false,HelpMessage="The CSV table header for student code (default is 'Student Code' for TTv7)")]
        [string]
        $username_header = "Student Code",
        [Parameter(Mandatory=$false,HelpMessage="The CSV table header for year level (default is 'Roll Class Code' for TTv7)")]
        [string]
        $group_header = "Roll Class Code",
        [Parameter(Mandatory=$false,HelpMessage="Prepends this to the beggining of each year level (eg. Year -> Year7)")]
        [string]
        $prefix = "",
        [Parameter(Mandatory=$false,HelpMessage="Appends this to the beggining of each year level (eg. students -> 7students)")]
        [string]
        $postfix = "",
        [Parameter(Mandatory=$false,HelpMessage="Strips leading zeros on group year")]
        [switch]
        $stripzeros,
        [Parameter(Mandatory=$false,HelpMessage="Selects only numbers from the group field (eg. '7B' -> '7')")]
        [switch]
        $onlynumbers,
        [Parameter(Mandatory=$false,HelpMessage='Filters records in CSV according to this filter (eg. -filter {$_.STATUS -eq "ACTV"')]
        [scriptblock]
        $filter
    )
    Begin{
        $years = @{}
        if ($onlynumbers) {
            if ($stripzeros) {
                Import-Csv $csvfile |Where-Object $filter |ForEach-Object {$years.$($($_.$group_header -replace '\D+').trimstart('0')) += @($($_.$username_header))}
            } else {
                Import-Csv $csvfile |Where-Object $filter |ForEach-Object {$years.$($_.$group_header -replace '\D+') += @($($_.$username_header))}
            }
        } else {
            if ($stripzeros) {
                Import-Csv $csvfile |Where-Object $filter |ForEach-Object {$years.$($($_.$group_header).trimstart('0')) += @($($_.$username_header))}
            } else {
                Import-Csv $csvfile |Where-Object $filter |ForEach-Object {$years.$_.$group_header += @($($_.$username_header))}
            }
        }
    }
    Process{
        foreach ($group in $years.Keys) {
            $unique_users = @()
            $adusers = @()
            $unique_users += $years.$group |Select-Object -Unique
            foreach ($user in $unique_users) {
                try {
                    $adusers += $(Get-ADUser -Identity $user)
                }
                catch {
                    Write-Warning "Couldn't find AD User account for $user"
                }
            }
            $non_members = Get-ADUser -Filter * |Where-Object {$_.samaccountname -notin $unique_users}
            Write-Debug "Group: $group"
            Write-Debug "Members: $adusers"
            Write-Debug "Non-members: $non_members"
            $props = @{
                'Identity' = "$prefix$group$postfix";
                'Members' = $adusers;
                'NonMembers' = $non_members
            }
            $obj = New-Object -TypeName PSObject -Property $props
            $obj.PSObject.TypeNames.Insert(0,"ST.GroupUsers")
            Write-Output $obj
        }
    }
}