ReportCommands/ComplianceDetailReports.ps1
|
#requires -Version 5.1 function Get-KeeperComplianceManagedUserEmailSet { $enterprise = getEnterprise $set = [System.Collections.Generic.HashSet[string]]::new([System.StringComparer]::OrdinalIgnoreCase) foreach ($eu in $enterprise.enterpriseData.Users) { if ($eu.UserStatus -eq [KeeperSecurity.Enterprise.UserStatus]::Inactive) { continue } if ($eu.Email) { $set.Add([string]$eu.Email) | Out-Null } } return $set } function Get-KeeperComplianceVaultRecordUidsForUser { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter(Mandatory = $true)][long]$UserUid ) $set = [System.Collections.Generic.HashSet[string]]::new([System.StringComparer]::Ordinal) if ($Snapshot.OwnedRecordsByUser.ContainsKey($UserUid)) { foreach ($r in $Snapshot.OwnedRecordsByUser[$UserUid]) { if ($r) { $set.Add([string]$r) | Out-Null } } } foreach ($recordUid in $Snapshot.Records.Keys) { $rec = $Snapshot.Records[$recordUid] if ($rec.UserPermissions.ContainsKey($UserUid)) { $set.Add([string]$recordUid) | Out-Null continue } foreach ($sfUid in $rec.SharedFolderUids) { $sfKey = [string]$sfUid if (-not $Snapshot.SharedFolders.ContainsKey($sfKey)) { continue } $sf = $Snapshot.SharedFolders[$sfKey] $allFolderUids = Get-KeeperComplianceSharedFolderAllUserUids -Snapshot $Snapshot -SharedFolder $sf if ($allFolderUids -contains $UserUid) { $set.Add([string]$recordUid) | Out-Null break } } } return $set } function Get-KeeperComplianceRecordOwnerEmailFromSnapshot { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter(Mandatory = $true)][string]$RecordUid ) foreach ($ownerUid in $Snapshot.OwnedRecordsByUser.Keys) { $owned = $Snapshot.OwnedRecordsByUser[$ownerUid] if ($owned -and $owned.Contains($RecordUid) -and $Snapshot.Users.ContainsKey([long]$ownerUid)) { return [string]$Snapshot.Users[[long]$ownerUid].Email } } return '' } function Get-KeeperVaultRecordMetadataFallback { <# When compliance/SOX snapshot has no decrypted title, type, or URL for a record UID, try the current session vault (admin's vault). Fills gaps when the same record exists there. #> param( [Parameter(Mandatory = $true)][string]$RecordUid ) try { $vault = getVault $rec = $null if (-not $vault.TryGetKeeperRecord($RecordUid, [ref]$rec)) { return $null } $title = [string]$rec.Title $url = '' $rtype = '' if ($rec -is [KeeperSecurity.Vault.PasswordRecord]) { $pr = [KeeperSecurity.Vault.PasswordRecord]$rec $rtype = 'login' if ($pr.Link) { $url = ([string]$pr.Link).TrimEnd('/') } } elseif ($rec -is [KeeperSecurity.Vault.TypedRecord]) { $tr = [KeeperSecurity.Vault.TypedRecord]$rec if ($tr.TypeName) { $rtype = $tr.TypeName } $urlField = $null if ([KeeperSecurity.Vault.VaultDataExtensions]::FindTypedField($tr, 'url', $null, [ref]$urlField)) { $urlVal = [KeeperSecurity.Vault.VaultDataExtensions]::GetExternalValue($urlField) if ($urlVal) { $url = ([string]$urlVal).TrimEnd('/') } } } else { $tn = $rec.GetType().Name if ($tn -and $tn -ne 'KeeperRecord') { $rtype = $tn -replace 'Record$', '' } } return [PSCustomObject]@{ Title = $title RecordType = $rtype Url = $url } } catch { return $null } } function Get-KeeperRecordAccessAuditEventsForUser { param( [Parameter(Mandatory = $true)]$Auth, [Parameter(Mandatory = $true)][string]$UserEmail, [Parameter()][string[]]$VaultRecordUids, [Parameter()][switch]$VaultMode, [Parameter(Mandatory = $true)][int]$Limit ) $result = @{} if ($VaultMode -and (-not $VaultRecordUids -or $VaultRecordUids.Count -eq 0)) { return $result } $createdMax = $null $remaining = $null if ($VaultMode) { $remaining = [System.Collections.Generic.HashSet[string]]::new([System.StringComparer]::Ordinal) foreach ($r in $VaultRecordUids) { if ($r) { $remaining.Add([string]$r) | Out-Null } } } while ($true) { $filter = New-Object KeeperSecurity.Enterprise.AuditLogCommands.ReportFilter $filter.Username = @($UserEmail) if ($VaultMode) { if ($remaining.Count -eq 0) { break } $filter.RecordUid = @($remaining | Sort-Object) } if ($null -ne $createdMax) { $cf = New-Object KeeperSecurity.Enterprise.AuditLogCommands.CreatedFilter $cf.Max = $createdMax $cf.ExcludeMax = $true $filter.Created = $cf } $rq = New-Object KeeperSecurity.Enterprise.AuditLogCommands.GetAuditEventReportsCommand $rq.Filter = $filter $rq.ReportType = 'span' $rq.Aggregate = @('last_created') $rq.Columns = @('record_uid', 'ip_address', 'keeper_version') $rq.Order = 'descending' $rq.Limit = $Limit try { $rs = $Auth.ExecuteAuthCommand( $rq, [KeeperSecurity.Enterprise.AuditLogCommands.GetAuditEventReportsResponse], $true ).GetAwaiter().GetResult() } catch { Write-Warning "Record-access audit request failed for ${UserEmail}: $($_.Exception.Message)" break } $events = if ($rs -and $rs.Events) { @($rs.Events | Where-Object { $null -ne $_ }) } else { @() } if ($events.Count -eq 0) { break } foreach ($evt in $events) { $rUid = Get-KeeperComplianceAuditEventValue -Event $evt -Key 'record_uid' if (-not $rUid) { continue } $rUidStr = [string]$rUid if (-not $result.ContainsKey($rUidStr)) { $result[$rUidStr] = $evt } if ($null -ne $remaining) { $remaining.Remove($rUidStr) | Out-Null } } $lastEvt = $events[$events.Count - 1] if ($null -eq $lastEvt) { break } $lc = Get-KeeperComplianceAuditEventValue -Event $lastEvt -Key 'last_created' $lastCreatedEpoch = 0L if ($null -ne $lc) { [void][long]::TryParse($lc.ToString(), [ref]$lastCreatedEpoch) } if (($events.Count -lt $Limit) -or ($VaultMode -and $remaining.Count -eq 0) -or ($lastCreatedEpoch -le 0)) { break } $createdMax = $lastCreatedEpoch } return $result } function Test-KeeperRecordAccessRowPattern { param( [Parameter(Mandatory = $true)]$Row, [Parameter(Mandatory = $true)][string[]]$Patterns, [Parameter()][switch]$UseRegex ) $text = ($Row.PSObject.Properties | ForEach-Object { "$($_.Value)" }) -join "`t" foreach ($p in $Patterns) { if ([string]::IsNullOrWhiteSpace($p)) { continue } if ($UseRegex) { try { if ($text -match $p) { return $true } } catch { } } else { foreach ($prop in $Row.PSObject.Properties) { $v = $prop.Value if ($null -eq $v) { continue } $s = [string]$v if ($s -like $p) { return $true } } } } return $false } function ConvertTo-KeeperRecordAccessDisplayRows { param( [Parameter(Mandatory = $true)]$Rows, [Parameter()][ValidateSet('table', 'json', 'csv')][string]$Format = 'table' ) if ($Format -ne 'table' -or $Rows.Count -eq 0) { return $Rows } $lastOwner = [string]::Empty $out = [System.Collections.Generic.List[object]]::new() foreach ($r in $Rows) { $vo = [string]$r.vault_owner $showVo = $vo if ($vo -eq $lastOwner) { $showVo = '' } else { $lastOwner = $vo } $copy = [ordered]@{} foreach ($prop in $r.PSObject.Properties) { if ($prop.Name -eq 'vault_owner') { $copy[$prop.Name] = $showVo } else { $copy[$prop.Name] = $prop.Value } } $out.Add([PSCustomObject]$copy) | Out-Null } return @($out) } function Get-KeeperComplianceRecordAccessReport { <# .Synopsis Run record-access report .Parameter Email User email(s), enterprise user ID, or '@all' .Parameter ReportType 'history' (default) or 'vault' .Parameter Format table (default), json, or csv .Parameter Output File path for json/csv output .Parameter Node Filter by node .Parameter Username Filter by username .Parameter Team Filter by team .Parameter Pattern Wildcard filter strings .Parameter PatternRegex Regex filter (mutually exclusive with -Pattern) #> [CmdletBinding(DefaultParameterSetName = 'Default')] param( [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [string[]]$Email, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [ValidateSet('history', 'vault')][string]$ReportType = 'history', [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [ValidateSet('table', 'json', 'csv')][string]$Format = 'table', [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [string]$Output, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [string]$Node, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [string[]]$Username, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [string[]]$Team, [Parameter(ParameterSetName = 'Default')][string[]]$Pattern, [Parameter(ParameterSetName = 'RegexPatterns')][string[]]$PatternRegex, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [switch]$Rebuild, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [switch]$NoRebuild, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [switch]$NoCache, [Parameter(ParameterSetName = 'Default')] [Parameter(ParameterSetName = 'RegexPatterns')] [switch]$Aging ) $apiRowLimit = 2000 Invoke-KeeperComplianceReportSession -NoCache:$NoCache -ScriptBlock { Write-KeeperComplianceStatus "Starting compliance record-access-report. ReportType=$ReportType Format=$Format Rebuild=$Rebuild NoRebuild=$NoRebuild NoCache=$NoCache Aging=$Aging." $enterprise = getEnterprise $auth = $enterprise.loader.Auth $managedSet = Get-KeeperComplianceManagedUserEmailSet $allowedUserIds = $null $fetchIds = Resolve-KeeperComplianceFetchOwnerIds -Username $Username -Team $Team -Node $Node if ($null -ne $fetchIds) { $allowedUserIds = [System.Collections.Generic.HashSet[long]]::new() foreach ($id in $fetchIds) { $allowedUserIds.Add([long]$id) | Out-Null } } if ($null -eq $fetchIds) { Write-KeeperComplianceStatus "Record-access owner pre-filter: all enterprise users (no Node/Username/Team filter)." } elseif ($fetchIds.Count -eq 0) { Write-KeeperComplianceStatus "Record-access owner pre-filter: 0 user(s) matched (Node/Username/Team exclude everyone)." } else { Write-KeeperComplianceStatus "Record-access owner pre-filter matched $($fetchIds.Count) user(s)." } $emailArgs = $Email if (-not $emailArgs -or $emailArgs.Count -eq 0) { $emailArgs = @('@all') } $resolvedEmails = [System.Collections.Generic.List[string]]::new() foreach ($ref in $emailArgs) { if ($ref -ieq '@all') { $sortedUsers = @($enterprise.enterpriseData.Users | Sort-Object Email) foreach ($eu in $sortedUsers) { if ($eu.UserStatus -eq [KeeperSecurity.Enterprise.UserStatus]::Inactive) { continue } if (-not $eu.Email) { continue } if ($null -ne $allowedUserIds -and -not $allowedUserIds.Contains([long]$eu.Id)) { continue } $resolvedEmails.Add([string]$eu.Email) | Out-Null } continue } $trim = $ref.Trim() if ($trim -match '^\d+$') { $eu = $null if ($enterprise.enterpriseData.TryGetUserById([long]$trim, [ref]$eu) -and $eu -and $eu.Email) { if ($null -ne $allowedUserIds -and -not $allowedUserIds.Contains([long]$eu.Id)) { continue } $resolvedEmails.Add([string]$eu.Email) | Out-Null } continue } if (-not $managedSet.Contains($trim)) { continue } $eu = $null if (-not $enterprise.enterpriseData.TryGetUserByEmail($trim, [ref]$eu) -or -not $eu) { continue } if ($null -ne $allowedUserIds -and -not $allowedUserIds.Contains([long]$eu.Id)) { continue } $resolvedEmails.Add($trim) | Out-Null } $seen = @{} $targetEmails = [System.Collections.Generic.List[string]]::new() foreach ($e in $resolvedEmails) { $k = $e.ToLowerInvariant() if ($seen[$k]) { continue } $seen[$k] = $true $targetEmails.Add($e) | Out-Null } if ($targetEmails.Count -eq 0) { Write-Host "No users selected for record-access report." return } $snapshot = Get-KeeperComplianceSnapshot -Rebuild:$Rebuild -NoRebuild:$NoRebuild -OwnerUserIds $null $rows = [System.Collections.Generic.List[object]]::new() $vaultMode = ($ReportType -eq 'vault') foreach ($userEmail in $targetEmails) { $eu = $null if (-not $enterprise.enterpriseData.TryGetUserByEmail($userEmail, [ref]$eu) -or -not $eu) { continue } $userUid = [long]$eu.Id $vaultUids = $null if ($vaultMode) { $vaultSet = Get-KeeperComplianceVaultRecordUidsForUser -Snapshot $snapshot -UserUid $userUid $vaultUids = @($vaultSet) } $auditMap = Get-KeeperRecordAccessAuditEventsForUser -Auth $auth -UserEmail $userEmail -VaultRecordUids $vaultUids ` -VaultMode:$vaultMode -Limit $apiRowLimit $recordUids = [System.Collections.Generic.List[string]]::new() if ($vaultMode) { foreach ($u in $vaultUids) { $recordUids.Add([string]$u) | Out-Null } } else { foreach ($k in $auditMap.Keys) { $recordUids.Add([string]$k) | Out-Null } } foreach ($recUid in $recordUids) { $evt = $null if ($auditMap.ContainsKey([string]$recUid)) { $evt = $auditMap[[string]$recUid] } $rec = $null if ($snapshot.Records.ContainsKey([string]$recUid)) { $rec = $snapshot.Records[[string]$recUid] } $title = if ($rec) { [string]$rec.Title } else { '' } $rtype = if ($rec) { [string]$rec.RecordType } else { '' } $url = if ($rec -and $rec.Url) { ([string]$rec.Url).TrimEnd('/') } else { '' } if ([string]::IsNullOrWhiteSpace($title) -or [string]::IsNullOrWhiteSpace($rtype) -or [string]::IsNullOrWhiteSpace($url)) { $vaultMeta = Get-KeeperVaultRecordMetadataFallback -RecordUid $recUid if ($vaultMeta) { if ([string]::IsNullOrWhiteSpace($title) -and $vaultMeta.Title) { $title = [string]$vaultMeta.Title } if ([string]::IsNullOrWhiteSpace($rtype) -and $vaultMeta.RecordType) { $rtype = [string]$vaultMeta.RecordType } if ([string]::IsNullOrWhiteSpace($url) -and $vaultMeta.Url) { $url = [string]$vaultMeta.Url } } } $inTrash = if ($rec) { [bool]$rec.InTrash } else { $false } $ip = '' $device = '' $lastAccess = $null if ($evt) { $ip = [string](Get-KeeperComplianceAuditEventValue -Event $evt -Key 'ip_address') $device = [string](Get-KeeperComplianceAuditEventValue -Event $evt -Key 'keeper_version') $lc = Get-KeeperComplianceAuditEventValue -Event $evt -Key 'last_created' $lastAccess = ConvertTo-KeeperComplianceDateTime -EpochValue $lc } $ownerEmail = Get-KeeperComplianceRecordOwnerEmailFromSnapshot -Snapshot $snapshot -RecordUid $recUid $row = [ordered]@{ vault_owner = $userEmail record_uid = $recUid record_title = $title record_type = $rtype record_url = $url has_attachments = $false in_trash = $inTrash record_owner = $ownerEmail ip_address = $ip device = $device last_access = $lastAccess } $rows.Add([PSCustomObject]$row) | Out-Null } } $reportRows = @($rows) if ($Pattern -and $Pattern.Count -gt 0) { $reportRows = @( $reportRows | Where-Object { Test-KeeperRecordAccessRowPattern -Row $_ -Patterns $Pattern -UseRegex:$false } ) } elseif ($PatternRegex -and $PatternRegex.Count -gt 0) { $reportRows = @( $reportRows | Where-Object { Test-KeeperRecordAccessRowPattern -Row $_ -Patterns $PatternRegex -UseRegex:$true } ) } if ($Aging -and $reportRows.Count -gt 0) { $agingUids = @($reportRows | ForEach-Object { [string]$_.record_uid } | Where-Object { $_ } | Sort-Object -Unique) Write-KeeperComplianceStatus "Applying aging to $($agingUids.Count) unique record(s)." $agingData = Get-KeeperComplianceAgingData -RecordUids $agingUids $newRows = [System.Collections.Generic.List[object]]::new() foreach ($r in $reportRows) { $uidKey = [string]$r.record_uid $ag = $null if ($agingData -and $agingData.ContainsKey($uidKey)) { $ag = $agingData[$uidKey] } $nr = [ordered]@{} foreach ($p in $r.PSObject.Properties) { $nr[$p.Name] = $p.Value } if ($ag) { $nr['created'] = $ag['created'] $nr['last_pw_change'] = $ag['last_pw_change'] $nr['last_modified'] = $ag['last_modified'] $nr['last_rotation'] = $ag['last_rotation'] } else { $nr['created'] = $null $nr['last_pw_change'] = $null $nr['last_modified'] = $null $nr['last_rotation'] = $null } $newRows.Add([PSCustomObject]$nr) | Out-Null } $reportRows = @($newRows) } if ($reportRows.Count -eq 0) { Write-KeeperComplianceStatus "No record-access rows matched." Write-Host "No compliance record-access report rows found." return } $displayRows = ConvertTo-KeeperRecordAccessDisplayRows -Rows $reportRows -Format $Format Write-KeeperComplianceStatus "Rendering $($reportRows.Count) row(s) as $Format." $tableCols = [System.Collections.Generic.List[string]]::new() foreach ($c in @( 'vault_owner', 'record_uid', 'record_title', 'record_type', 'record_url', 'has_attachments', 'in_trash', 'record_owner', 'ip_address', 'device', 'last_access' )) { $tableCols.Add($c) | Out-Null } if ($Aging) { foreach ($c in @('created', 'last_pw_change', 'last_modified', 'last_rotation')) { $tableCols.Add($c) | Out-Null } } Write-KeeperReportOutput -Rows $reportRows -DisplayRows $displayRows -Format $Format -Output $Output -JsonDepth 8 ` -TableColumns @($tableCols) } } New-Alias -Name record-access-report -Value Get-KeeperComplianceRecordAccessReport function Get-KeeperComplianceTeamReportFilters { param( [Parameter()][string[]]$Team ) $enterprise = getEnterprise $enterpriseData = $enterprise.enterpriseData $teamUids = [System.Collections.Generic.HashSet[string]]::new() if (Test-KeeperComplianceHasNonEmptyStringList -Strings $Team) { foreach ($teamRef in $Team) { if ([string]::IsNullOrWhiteSpace([string]$teamRef)) { continue } $resolvedTeam = Get-KeeperTeamByNameOrUid -EnterpriseData $enterpriseData -TeamInput $teamRef if (-not $resolvedTeam) { Write-Warning "No enterprise team matched '$teamRef' for compliance team filter." continue } $teamUids.Add([string]$resolvedTeam.Uid) | Out-Null } } return [PSCustomObject]@{ TeamUids = if ($teamUids.Count -gt 0) { @($teamUids | Sort-Object) } else { $null } } } function Get-KeeperComplianceSharedFolderAllUserUids { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter(Mandatory = $true)]$SharedFolder ) $allUserUids = [System.Collections.Generic.HashSet[long]]::new() foreach ($userUid in $SharedFolder.Users) { $allUserUids.Add([long]$userUid) | Out-Null } $enterprise = getEnterprise $enterpriseData = $enterprise.enterpriseData foreach ($teamUid in $SharedFolder.Teams) { if ($Snapshot.Teams.ContainsKey([string]$teamUid)) { foreach ($teamUserUid in $Snapshot.Teams[[string]$teamUid].Users) { $allUserUids.Add([long]$teamUserUid) | Out-Null } } else { foreach ($teamUserUid in $enterpriseData.GetUsersForTeam([string]$teamUid)) { $allUserUids.Add([long]$teamUserUid) | Out-Null } } } return @($allUserUids | Sort-Object) } function Get-KeeperComplianceSharedFolderUserEmails { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter(Mandatory = $true)][string]$TeamUid ) $enterprise = getEnterprise $enterpriseData = $enterprise.enterpriseData $emails = [System.Collections.Generic.HashSet[string]]::new([System.StringComparer]::OrdinalIgnoreCase) $teamUserIds = @() if ($Snapshot.Teams.ContainsKey([string]$TeamUid)) { $teamUserIds = @($Snapshot.Teams[[string]$TeamUid].Users) } else { $teamUserIds = @($enterpriseData.GetUsersForTeam([string]$TeamUid)) } foreach ($userUid in $teamUserIds) { $email = $null if ($Snapshot.Users.ContainsKey([long]$userUid)) { $email = [string]$Snapshot.Users[[long]$userUid].Email } else { $enterpriseUser = $null if ($enterpriseData.TryGetUserById([long]$userUid, [ref]$enterpriseUser) -and $enterpriseUser) { $email = [string]$enterpriseUser.Email } } if ($email) { $emails.Add($email) | Out-Null } } return @($emails | Sort-Object) } function Get-KeeperComplianceSharedFolderNameLookup { $lookup = @{} try { [KeeperSecurity.Vault.VaultOnline]$vault = getVault foreach ($sharedFolder in $vault.SharedFolders) { if ($sharedFolder.Uid) { $lookup[[string]$sharedFolder.Uid] = [string]$sharedFolder.Name } } } catch { } return $lookup } function Get-KeeperComplianceTeamPermissionText { param( [Parameter(Mandatory = $true)]$Team ) $permissions = @() if (-not $Team.RestrictShare) { $permissions += 'Can Share' } if (-not $Team.RestrictEdit) { $permissions += 'Can Edit' } if ($permissions.Count -eq 0) { return 'Read Only' } return ($permissions -join '; ') } function Get-KeeperComplianceTeamReportRows { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter()][string[]]$Team, [Parameter()]$Node, [Parameter()][switch]$ShowTeamUsers ) $enterprise = getEnterprise $enterpriseData = $enterprise.enterpriseData $filterInfo = Get-KeeperComplianceTeamReportFilters -Team $Team $teamLookup = $null if ($null -ne $filterInfo.TeamUids) { $teamLookup = [System.Collections.Generic.HashSet[string]]::new([System.StringComparer]::OrdinalIgnoreCase) foreach ($teamUid in $filterInfo.TeamUids) { $teamLookup.Add([string]$teamUid) | Out-Null } } $filterTeamNodeSubtreeIds = $null $filterTeamNodeSkip = $false if (Test-KeeperComplianceHasNodeFilter -Node $Node) { $resolvedFilterNode = Resolve-KeeperComplianceNode -Node $Node.Trim() -Context 'compliance team report node filter' $filterTargetNodeId = [long]$resolvedFilterNode.Id $rootNodeId = [long]$enterpriseData.RootNode.Id if ($filterTargetNodeId -eq $rootNodeId) { $filterTeamNodeSkip = $true } else { $filterTeamNodeSubtreeIds = Get-KeeperComplianceEnterpriseNodeSubtreeIds -EnterpriseData $enterpriseData -RootNodeId $filterTargetNodeId } } $sharedFolderNames = Get-KeeperComplianceSharedFolderNameLookup $rows = [System.Collections.Generic.List[PSCustomObject]]::new() foreach ($folderEntry in ($Snapshot.SharedFolders.Values | Sort-Object Uid)) { $folderRecordUids = @($folderEntry.RecordPermissions.Keys) if ($folderRecordUids.Count -le 0) { continue } $recordCount = @($folderRecordUids).Count if ($teamLookup) { $matchesTeam = $false foreach ($tUid in $folderEntry.Teams) { if ($teamLookup.Contains([string]$tUid)) { $matchesTeam = $true break } } if (-not $matchesTeam) { continue } } foreach ($teamUid in (@($folderEntry.Teams) | Sort-Object)) { if ($teamLookup -and -not $teamLookup.Contains([string]$teamUid)) { continue } $teamObject = $null if (-not $enterpriseData.TryGetTeam([string]$teamUid, [ref]$teamObject) -or -not $teamObject) { continue } $teamNodeId = [long]$teamObject.ParentNodeId if ($teamNodeId -le 0) { $teamNodeId = [long]$enterpriseData.RootNode.Id } if ($Node -and -not $filterTeamNodeSkip) { if ($null -eq $filterTeamNodeSubtreeIds -or $filterTeamNodeSubtreeIds.Count -eq 0 -or -not $filterTeamNodeSubtreeIds.ContainsKey("$([long]$teamNodeId)")) { continue } } $teamNodePath = Get-KeeperNodePath -NodeId $teamNodeId -OmitRoot $row = [ordered]@{ team_name = [string]$teamObject.Name team_uid = [string]$teamUid node = [string]$teamNodePath shared_folder_name = if ($sharedFolderNames.ContainsKey([string]$folderEntry.Uid)) { [string]$sharedFolderNames[[string]$folderEntry.Uid] } else { '' } shared_folder_uid = [string]$folderEntry.Uid permissions = Get-KeeperComplianceTeamPermissionText -Team $teamObject records = [int]$recordCount } if ($ShowTeamUsers) { $row['team_users'] = Get-KeeperComplianceSharedFolderUserEmails -Snapshot $Snapshot -TeamUid ([string]$teamUid) } $rows.Add([PSCustomObject]$row) | Out-Null } } return @($rows | Sort-Object shared_folder_uid, team_name) } function Get-KeeperComplianceTeamReport { <# .Synopsis Run compliance team report #> [CmdletBinding()] param( [Parameter()][ValidateSet('table', 'json', 'csv')][string]$Format = 'table', [Parameter()][string]$Output, [Parameter()][string]$Node, [Parameter()][string[]]$Team, [Parameter()][switch]$ShowTeamUsers, [Parameter()][switch]$Rebuild, [Parameter()][switch]$NoRebuild, [Parameter()][switch]$NoCache ) $reportRows = Invoke-KeeperComplianceReportSession -NoCache:$NoCache -ScriptBlock { Write-KeeperComplianceStatus "Starting compliance-team-report. Format=$Format Rebuild=$Rebuild NoRebuild=$NoRebuild NoCache=$NoCache ShowTeamUsers=$ShowTeamUsers." $fetchOwnerIds = Resolve-KeeperComplianceFetchOwnerIds -Node $Node if ((Test-KeeperComplianceHasNodeFilter -Node $Node) -and $null -ne $fetchOwnerIds -and $fetchOwnerIds.Count -eq 0) { Write-Warning "No enterprise users matched the provided node filter." } $ownerIdsForSnapshot = if (Test-KeeperComplianceHasNodeFilter -Node $Node) { $null } else { $fetchOwnerIds } $snapshot = Get-KeeperComplianceSnapshot -Rebuild:$Rebuild -NoRebuild:$NoRebuild -OwnerUserIds $ownerIdsForSnapshot -SharedOnly $reportRows = Get-KeeperComplianceTeamReportRows -Snapshot $snapshot -Team $Team ` -Node $Node -ShowTeamUsers:$ShowTeamUsers return ,@($reportRows) } if ($reportRows.Count -eq 0) { Write-Host "No compliance team report rows found." return } $displayRows = @( $reportRows | ForEach-Object { $row = [ordered]@{} foreach ($property in $_.PSObject.Properties) { if ($property.Name -eq 'team_users') { $row[$property.Name] = @($property.Value) -join ', ' } else { $row[$property.Name] = $property.Value } } [PSCustomObject]$row } ) Write-KeeperReportOutput -Rows $reportRows -DisplayRows $displayRows -Format $Format -Output $Output -JsonDepth 5 } New-Alias -Name compliance-team-report -Value Get-KeeperComplianceTeamReport function Get-KeeperComplianceSummaryStatsForUser { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter(Mandatory = $true)][long]$UserUid, [Parameter(Mandatory = $true)][string]$Email ) $vaultSet = Get-KeeperComplianceVaultRecordUidsForUser -Snapshot $Snapshot -UserUid $UserUid $totalItems = $vaultSet.Count $numOwned = 0 $activeOwned = 0 $deletedOwned = 0 if ($Snapshot.OwnedRecordsByUser.ContainsKey($UserUid)) { $ownedSet = $Snapshot.OwnedRecordsByUser[$UserUid] $numOwned = $ownedSet.Count foreach ($r in $ownedSet) { $rk = [string]$r $inTrash = $false if ($Snapshot.Records.ContainsKey($rk)) { $inTrash = [bool]$Snapshot.Records[$rk].InTrash } if ($inTrash) { $deletedOwned++ } else { $activeOwned++ } } } return [PSCustomObject]@{ email = $Email total_items = [int]$totalItems total_owned = [int]$numOwned active_owned = [int]$activeOwned deleted_owned = [int]$deletedOwned } } function Get-KeeperComplianceSummaryReportRows { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter()][string[]]$Team, [Parameter()]$Node ) $enterprise = getEnterprise $enterpriseData = $enterprise.enterpriseData $fetchIds = Resolve-KeeperComplianceFetchOwnerIds -Team $Team -Node $Node if ((Test-KeeperComplianceHasNodeFilter -Node $Node) -and $null -ne $fetchIds -and @($fetchIds).Count -eq 0) { Write-Warning "No enterprise users matched the provided node (and team) filter." } $fetchIdSet = $null if ($null -ne $fetchIds) { $fetchIdSet = [System.Collections.Generic.HashSet[long]]::new() foreach ($id in @($fetchIds)) { $fetchIdSet.Add([long]$id) | Out-Null } } $rows = [System.Collections.Generic.List[object]]::new() $soxUserIds = [System.Collections.Generic.HashSet[long]]::new() foreach ($k in $Snapshot.Users.Keys) { $soxUserIds.Add([long]$k) | Out-Null } foreach ($eu in $enterpriseData.Users) { if ($eu.UserStatus -eq [KeeperSecurity.Enterprise.UserStatus]::Inactive) { continue } if (-not $eu.Email) { continue } if ($null -ne $fetchIdSet -and -not $fetchIdSet.Contains([long]$eu.Id)) { continue } $uid = [long]$eu.Id $email = [string]$eu.Email if ($soxUserIds.Contains($uid)) { $rows.Add((Get-KeeperComplianceSummaryStatsForUser -Snapshot $Snapshot -UserUid $uid -Email $email)) | Out-Null } else { $rows.Add([PSCustomObject]@{ email = $email total_items = 0 total_owned = 0 active_owned = 0 deleted_owned = 0 }) | Out-Null } } $sortedRows = [System.Collections.Generic.List[object]]::new() foreach ($r in (@($rows) | Sort-Object email)) { $sortedRows.Add($r) | Out-Null } $sumOwned = 0L $sumActive = 0L $sumDeleted = 0L foreach ($dr in $sortedRows) { $sumOwned += [long]$dr.total_owned $sumActive += [long]$dr.active_owned $sumDeleted += [long]$dr.deleted_owned } $sortedRows.Add([PSCustomObject]@{ email = 'TOTAL' total_items = $null total_owned = [long]$sumOwned active_owned = [long]$sumActive deleted_owned = [long]$sumDeleted }) | Out-Null return @($sortedRows) } function Get-KeeperComplianceSummaryReport { <# .Synopsis Run compliance summary report #> [CmdletBinding()] param( [Parameter()][ValidateSet('table', 'json', 'csv')][string]$Format = 'table', [Parameter()][string]$Output, [Parameter()][string]$Node, [Parameter()][string[]]$Team, [Parameter()][switch]$Rebuild, [Parameter()][switch]$NoRebuild, [Parameter()][switch]$NoCache ) $reportRows = Invoke-KeeperComplianceReportSession -NoCache:$NoCache -ScriptBlock { Write-KeeperComplianceStatus "Starting compliance summary-report. Format=$Format Rebuild=$Rebuild NoRebuild=$NoRebuild NoCache=$NoCache." $snapshot = Get-KeeperComplianceSnapshot -Rebuild:$Rebuild -NoRebuild:$NoRebuild -OwnerUserIds $null $reportRows = Get-KeeperComplianceSummaryReportRows -Snapshot $snapshot -Team $Team -Node $Node return ,@($reportRows) } if ($reportRows.Count -eq 0) { Write-Host "No compliance summary report rows found." return } $displayRows = @( $reportRows | ForEach-Object { $row = [ordered]@{} foreach ($property in $_.PSObject.Properties) { $row[$property.Name] = $property.Value } [PSCustomObject]$row } ) Write-KeeperReportOutput -Rows $reportRows -DisplayRows $displayRows -Format $Format -Output $Output -JsonDepth 5 ` -TableColumns @('email', 'total_items', 'total_owned', 'active_owned', 'deleted_owned') } New-Alias -Name compliance-summary-report -Value Get-KeeperComplianceSummaryReport function Get-KeeperComplianceSharedFolderReportRows { param( [Parameter(Mandatory = $true)]$Snapshot, [Parameter()][string[]]$Team, [Parameter()][switch]$ShowTeamUsers, [Parameter()]$Node, [Parameter()][long[]]$NodeScopeUserIds ) $enterprise = getEnterprise $enterpriseData = $enterprise.enterpriseData $filterInfo = Get-KeeperComplianceTeamReportFilters -Team $Team $teamLookup = $null if ($null -ne $filterInfo.TeamUids) { $teamLookup = [System.Collections.Generic.HashSet[string]]::new([System.StringComparer]::OrdinalIgnoreCase) foreach ($teamUid in $filterInfo.TeamUids) { $teamLookup.Add([string]$teamUid) | Out-Null } } $nodeUserIdSet = $null $recordsOwnedByNodeUsers = $null $filterTeamNodeSubtreeIds = $null $rootNodeIdSf = [long]$enterpriseData.RootNode.Id if (Test-KeeperComplianceHasNodeFilter -Node $Node) { if ($null -ne $NodeScopeUserIds -and @($NodeScopeUserIds).Count -gt 0) { $nodeUserIdSet = [System.Collections.Generic.HashSet[long]]::new() foreach ($id in @($NodeScopeUserIds)) { $nodeUserIdSet.Add([long]$id) | Out-Null } $recordsOwnedByNodeUsers = [System.Collections.Generic.HashSet[string]]::new([System.StringComparer]::OrdinalIgnoreCase) foreach ($uid in $nodeUserIdSet) { if (-not $Snapshot.OwnedRecordsByUser.ContainsKey([long]$uid)) { continue } foreach ($r in $Snapshot.OwnedRecordsByUser[[long]$uid]) { $recordsOwnedByNodeUsers.Add([string]$r) | Out-Null } } } $resolvedFilterNode = Resolve-KeeperComplianceNode -Node $Node.Trim() -Context 'compliance shared-folder report node filter' $filterTargetNodeId = [long]$resolvedFilterNode.Id $subtreeRootId = if ($filterTargetNodeId -eq $rootNodeIdSf) { $rootNodeIdSf } else { $filterTargetNodeId } $filterTeamNodeSubtreeIds = Get-KeeperComplianceEnterpriseNodeSubtreeIds -EnterpriseData $enterpriseData -RootNodeId $subtreeRootId } $rows = [System.Collections.Generic.List[object]]::new() foreach ($folderEntry in ($Snapshot.SharedFolders.Values | Sort-Object { $_.Uid })) { $recordUids = @($folderEntry.RecordPermissions.Keys | Sort-Object) if ($recordUids.Count -eq 0) { continue } if (Test-KeeperComplianceHasNodeFilter -Node $Node) { $folderRelevantToNode = $false if ($null -ne $recordsOwnedByNodeUsers -and $recordsOwnedByNodeUsers.Count -gt 0) { foreach ($ru in $recordUids) { if ($recordsOwnedByNodeUsers.Contains([string]$ru)) { $folderRelevantToNode = $true break } } } if (-not $folderRelevantToNode -and $null -ne $nodeUserIdSet) { foreach ($userUid in $folderEntry.Users) { if ($nodeUserIdSet.Contains([long]$userUid)) { $folderRelevantToNode = $true break } } } if (-not $folderRelevantToNode) { foreach ($tuid in $folderEntry.Teams) { $teamObj = $null if ($enterpriseData.TryGetTeam([string]$tuid, [ref]$teamObj) -and $teamObj) { $teamHomeId = [long]$teamObj.ParentNodeId if ($teamHomeId -le 0) { $teamHomeId = $rootNodeIdSf } if ($null -ne $filterTeamNodeSubtreeIds -and $filterTeamNodeSubtreeIds.Count -gt 0 -and $filterTeamNodeSubtreeIds.ContainsKey("$([long]$teamHomeId)")) { $folderRelevantToNode = $true break } } if (-not $folderRelevantToNode -and $null -ne $nodeUserIdSet) { $teamUserIds = @() if ($Snapshot.Teams.ContainsKey([string]$tuid)) { $teamUserIds = @($Snapshot.Teams[[string]$tuid].Users) } else { $teamUserIds = @($enterpriseData.GetUsersForTeam([string]$tuid)) } foreach ($tu in $teamUserIds) { if ($nodeUserIdSet.Contains([long]$tu)) { $folderRelevantToNode = $true break } } } if ($folderRelevantToNode) { break } } } if (-not $folderRelevantToNode) { continue } } if ($teamLookup) { $matchesTeam = $false foreach ($t in $folderEntry.Teams) { if ($teamLookup.Contains([string]$t)) { $matchesTeam = $true break } } if (-not $matchesTeam) { continue } } $teamUids = @($folderEntry.Teams | Sort-Object) $teamNames = [System.Collections.Generic.List[string]]::new() $teamNodePaths = [System.Collections.Generic.List[string]]::new() $rootNodeIdForTeams = [long]$enterpriseData.RootNode.Id foreach ($tid in $teamUids) { $teamObj = $null if ($enterpriseData.TryGetTeam([string]$tid, [ref]$teamObj) -and $teamObj) { $teamNames.Add([string]$teamObj.Name) | Out-Null $teamNodeId = [long]$teamObj.ParentNodeId if ($teamNodeId -le 0) { $teamNodeId = $rootNodeIdForTeams } $teamNodePaths.Add([string](Get-KeeperNodePath -NodeId $teamNodeId -OmitRoot)) | Out-Null } else { $teamNames.Add('') | Out-Null $teamNodePaths.Add('') | Out-Null } } $emailParts = [System.Collections.Generic.List[string]]::new() if ($ShowTeamUsers) { foreach ($tid in $teamUids) { foreach ($em in Get-KeeperComplianceSharedFolderUserEmails -Snapshot $Snapshot -TeamUid ([string]$tid)) { $emailParts.Add("(TU)$em") | Out-Null } } } foreach ($userUid in ($folderEntry.Users | Sort-Object)) { if ($Snapshot.Users.ContainsKey([long]$userUid)) { $emailParts.Add([string]$Snapshot.Users[[long]$userUid].Email) | Out-Null } } $recordTitles = [System.Collections.Generic.List[string]]::new() foreach ($ru in $recordUids) { $rt = '' if ($Snapshot.Records.ContainsKey([string]$ru)) { $rt = [string]$Snapshot.Records[[string]$ru].Title } $recordTitles.Add($rt) | Out-Null } $rows.Add([PSCustomObject][ordered]@{ shared_folder_uid = [string]$folderEntry.Uid team_uid = @($teamUids) -join ', ' team_name = @($teamNames) -join ', ' node = @($teamNodePaths) -join ', ' record_uid = @($recordUids) -join ', ' record_title = @($recordTitles) -join ', ' email = @($emailParts) -join ', ' }) | Out-Null } return @($rows | Sort-Object shared_folder_uid) } function Get-KeeperComplianceSharedFolderReport { <# .Synopsis Run compliance shared-folder report .Parameter ShowTeamUsers Include team members in the email column #> [CmdletBinding()] param( [Parameter()][ValidateSet('table', 'json', 'csv')][string]$Format = 'table', [Parameter()][string]$Output, [Parameter()][string]$Node, [Parameter()][string[]]$Team, [Parameter()][switch]$ShowTeamUsers, [Parameter()][switch]$Rebuild, [Parameter()][switch]$NoRebuild, [Parameter()][switch]$NoCache ) $reportRows = Invoke-KeeperComplianceReportSession -NoCache:$NoCache -ScriptBlock { Write-KeeperComplianceStatus "Starting compliance shared-folder-report. Format=$Format Rebuild=$Rebuild NoRebuild=$NoRebuild NoCache=$NoCache ShowTeamUsers=$ShowTeamUsers." $fetchOwnerIds = Resolve-KeeperComplianceFetchOwnerIds -Node $Node if ((Test-KeeperComplianceHasNodeFilter -Node $Node) -and $null -ne $fetchOwnerIds -and @($fetchOwnerIds).Count -eq 0) { Write-Warning "No enterprise users in the node subtree for user/record checks; folders may still match via team home node." } $ownerIdsForSnapshot = if (Test-KeeperComplianceHasNodeFilter -Node $Node) { $null } else { $fetchOwnerIds } $snapshot = Get-KeeperComplianceSnapshot -Rebuild:$Rebuild -NoRebuild:$NoRebuild -OwnerUserIds $ownerIdsForSnapshot -SharedOnly $reportRows = Get-KeeperComplianceSharedFolderReportRows -Snapshot $snapshot -Team $Team ` -ShowTeamUsers:$ShowTeamUsers -Node $Node -NodeScopeUserIds $fetchOwnerIds return ,@($reportRows) } if ($ShowTeamUsers) { Write-Host "(TU) denotes a user whose membership in a team grants them access to the shared folder." -ForegroundColor DarkGray } if ($reportRows.Count -eq 0) { Write-Host "No compliance shared-folder report rows found." return } Write-KeeperReportOutput -Rows $reportRows -DisplayRows $reportRows -Format $Format -Output $Output -JsonDepth 6 ` -TableColumns @('shared_folder_uid', 'team_uid', 'team_name', 'node', 'record_uid', 'record_title', 'email') } New-Alias -Name compliance-shared-folder-report -Value Get-KeeperComplianceSharedFolderReport # SIG # Begin signature block # MIInvgYJKoZIhvcNAQcCoIInrzCCJ6sCAQExDzANBglghkgBZQMEAgEFADB5Bgor # BgEEAYI3AgEEoGswaTA0BgorBgEEAYI3AgEeMCYCAwEAAAQQH8w7YFlLCE63JNLG # KX7zUQIBAAIBAAIBAAIBAAIBADAxMA0GCWCGSAFlAwQCAQUABCC0aK7Ls4bxpIbQ # f9MUw29dCW5usLir+ibMCfvEibA1R6CCITswggWNMIIEdaADAgECAhAOmxiO+dAt # 5+/bUOIIQBhaMA0GCSqGSIb3DQEBDAUAMGUxCzAJBgNVBAYTAlVTMRUwEwYDVQQK # EwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xJDAiBgNV # BAMTG0RpZ2lDZXJ0IEFzc3VyZWQgSUQgUm9vdCBDQTAeFw0yMjA4MDEwMDAwMDBa # Fw0zMTExMDkyMzU5NTlaMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2Vy # dCBJbmMxGTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lD # ZXJ0IFRydXN0ZWQgUm9vdCBHNDCCAiIwDQYJKoZIhvcNAQEBBQADggIPADCCAgoC # ggIBAL/mkHNo3rvkXUo8MCIwaTPswqclLskhPfKK2FnC4SmnPVirdprNrnsbhA3E # MB/zG6Q4FutWxpdtHauyefLKEdLkX9YFPFIPUh/GnhWlfr6fqVcWWVVyr2iTcMKy # unWZanMylNEQRBAu34LzB4TmdDttceItDBvuINXJIB1jKS3O7F5OyJP4IWGbNOsF # xl7sWxq868nPzaw0QF+xembud8hIqGZXV59UWI4MK7dPpzDZVu7Ke13jrclPXuU1 # 5zHL2pNe3I6PgNq2kZhAkHnDeMe2scS1ahg4AxCN2NQ3pC4FfYj1gj4QkXCrVYJB # MtfbBHMqbpEBfCFM1LyuGwN1XXhm2ToxRJozQL8I11pJpMLmqaBn3aQnvKFPObUR # WBf3JFxGj2T3wWmIdph2PVldQnaHiZdpekjw4KISG2aadMreSx7nDmOu5tTvkpI6 # nj3cAORFJYm2mkQZK37AlLTSYW3rM9nF30sEAMx9HJXDj/chsrIRt7t/8tWMcCxB # YKqxYxhElRp2Yn72gLD76GSmM9GJB+G9t+ZDpBi4pncB4Q+UDCEdslQpJYls5Q5S # UUd0viastkF13nqsX40/ybzTQRESW+UQUOsxxcpyFiIJ33xMdT9j7CFfxCBRa2+x # q4aLT8LWRV+dIPyhHsXAj6KxfgommfXkaS+YHS312amyHeUbAgMBAAGjggE6MIIB # NjAPBgNVHRMBAf8EBTADAQH/MB0GA1UdDgQWBBTs1+OC0nFdZEzfLmc/57qYrhwP # TzAfBgNVHSMEGDAWgBRF66Kv9JLLgjEtUYunpyGd823IDzAOBgNVHQ8BAf8EBAMC # AYYweQYIKwYBBQUHAQEEbTBrMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdp # Y2VydC5jb20wQwYIKwYBBQUHMAKGN2h0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNv # bS9EaWdpQ2VydEFzc3VyZWRJRFJvb3RDQS5jcnQwRQYDVR0fBD4wPDA6oDigNoY0 # aHR0cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0QXNzdXJlZElEUm9vdENB # LmNybDARBgNVHSAECjAIMAYGBFUdIAAwDQYJKoZIhvcNAQEMBQADggEBAHCgv0Nc # Vec4X6CjdBs9thbX979XB72arKGHLOyFXqkauyL4hxppVCLtpIh3bb0aFPQTSnov # Lbc47/T/gLn4offyct4kvFIDyE7QKt76LVbP+fT3rDB6mouyXtTP0UNEm0Mh65Zy # oUi0mcudT6cGAxN3J0TU53/oWajwvy8LpunyNDzs9wPHh6jSTEAZNUZqaVSwuKFW # juyk1T3osdz9HNj0d1pcVIxv76FQPfx2CWiEn2/K2yCNNWAcAgPLILCsWKAOQGPF # mCLBsln1VWvPJ6tsds5vIy30fnFqI2si/xK4VC0nftg62fC2h5b9W9FcrBjDTZ9z # twGpn1eqXijiuZQwggawMIIEmKADAgECAhAIrUCyYNKcTJ9ezam9k67ZMA0GCSqG # SIb3DQEBDAUAMGIxCzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMx # GTAXBgNVBAsTEHd3dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lDZXJ0IFRy # dXN0ZWQgUm9vdCBHNDAeFw0yMTA0MjkwMDAwMDBaFw0zNjA0MjgyMzU5NTlaMGkx # CzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UEAxM4 # RGlnaUNlcnQgVHJ1c3RlZCBHNCBDb2RlIFNpZ25pbmcgUlNBNDA5NiBTSEEzODQg # MjAyMSBDQTEwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQDVtC9C0Cit # eLdd1TlZG7GIQvUzjOs9gZdwxbvEhSYwn6SOaNhc9es0JAfhS0/TeEP0F9ce2vnS # 1WcaUk8OoVf8iJnBkcyBAz5NcCRks43iCH00fUyAVxJrQ5qZ8sU7H/Lvy0daE6ZM # swEgJfMQ04uy+wjwiuCdCcBlp/qYgEk1hz1RGeiQIXhFLqGfLOEYwhrMxe6TSXBC # Mo/7xuoc82VokaJNTIIRSFJo3hC9FFdd6BgTZcV/sk+FLEikVoQ11vkunKoAFdE3 # /hoGlMJ8yOobMubKwvSnowMOdKWvObarYBLj6Na59zHh3K3kGKDYwSNHR7OhD26j # q22YBoMbt2pnLdK9RBqSEIGPsDsJ18ebMlrC/2pgVItJwZPt4bRc4G/rJvmM1bL5 # OBDm6s6R9b7T+2+TYTRcvJNFKIM2KmYoX7BzzosmJQayg9Rc9hUZTO1i4F4z8ujo # 7AqnsAMrkbI2eb73rQgedaZlzLvjSFDzd5Ea/ttQokbIYViY9XwCFjyDKK05huzU # tw1T0PhH5nUwjewwk3YUpltLXXRhTT8SkXbev1jLchApQfDVxW0mdmgRQRNYmtwm # KwH0iU1Z23jPgUo+QEdfyYFQc4UQIyFZYIpkVMHMIRroOBl8ZhzNeDhFMJlP/2NP # TLuqDQhTQXxYPUez+rbsjDIJAsxsPAxWEQIDAQABo4IBWTCCAVUwEgYDVR0TAQH/ # BAgwBgEB/wIBADAdBgNVHQ4EFgQUaDfg67Y7+F8Rhvv+YXsIiGX0TkIwHwYDVR0j # BBgwFoAU7NfjgtJxXWRM3y5nP+e6mK4cD08wDgYDVR0PAQH/BAQDAgGGMBMGA1Ud # JQQMMAoGCCsGAQUFBwMDMHcGCCsGAQUFBwEBBGswaTAkBggrBgEFBQcwAYYYaHR0 # cDovL29jc3AuZGlnaWNlcnQuY29tMEEGCCsGAQUFBzAChjVodHRwOi8vY2FjZXJ0 # cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNydDBDBgNVHR8E # PDA6MDigNqA0hjJodHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVz # dGVkUm9vdEc0LmNybDAcBgNVHSAEFTATMAcGBWeBDAEDMAgGBmeBDAEEATANBgkq # hkiG9w0BAQwFAAOCAgEAOiNEPY0Idu6PvDqZ01bgAhql+Eg08yy25nRm95RysQDK # r2wwJxMSnpBEn0v9nqN8JtU3vDpdSG2V1T9J9Ce7FoFFUP2cvbaF4HZ+N3HLIvda # qpDP9ZNq4+sg0dVQeYiaiorBtr2hSBh+3NiAGhEZGM1hmYFW9snjdufE5BtfQ/g+ # lP92OT2e1JnPSt0o618moZVYSNUa/tcnP/2Q0XaG3RywYFzzDaju4ImhvTnhOE7a # brs2nfvlIVNaw8rpavGiPttDuDPITzgUkpn13c5UbdldAhQfQDN8A+KVssIhdXNS # y0bYxDQcoqVLjc1vdjcshT8azibpGL6QB7BDf5WIIIJw8MzK7/0pNVwfiThV9zeK # iwmhywvpMRr/LhlcOXHhvpynCgbWJme3kuZOX956rEnPLqR0kq3bPKSchh/jwVYb # KyP/j7XqiHtwa+aguv06P0WmxOgWkVKLQcBIhEuWTatEQOON8BUozu3xGFYHKi8Q # xAwIZDwzj64ojDzLj4gLDb879M4ee47vtevLt/B3E+bnKD+sEq6lLyJsQfmCXBVm # zGwOysWGw/YmMwwHS6DTBwJqakAwSEs0qFEgu60bhQjiWQ1tygVQK+pKHJ6l/aCn # HwZ05/LWUpD9r4VIIflXO7ScA+2GRfS0YW6/aOImYIbqyK+p/pQd52MbOoZWeE4w # gga0MIIEnKADAgECAhANx6xXBf8hmS5AQyIMOkmGMA0GCSqGSIb3DQEBCwUAMGIx # CzAJBgNVBAYTAlVTMRUwEwYDVQQKEwxEaWdpQ2VydCBJbmMxGTAXBgNVBAsTEHd3 # dy5kaWdpY2VydC5jb20xITAfBgNVBAMTGERpZ2lDZXJ0IFRydXN0ZWQgUm9vdCBH # NDAeFw0yNTA1MDcwMDAwMDBaFw0zODAxMTQyMzU5NTlaMGkxCzAJBgNVBAYTAlVT # MRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjFBMD8GA1UEAxM4RGlnaUNlcnQgVHJ1 # c3RlZCBHNCBUaW1lU3RhbXBpbmcgUlNBNDA5NiBTSEEyNTYgMjAyNSBDQTEwggIi # MA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQC0eDHTCphBcr48RsAcrHXbo0Zo # dLRRF51NrY0NlLWZloMsVO1DahGPNRcybEKq+RuwOnPhof6pvF4uGjwjqNjfEvUi # 6wuim5bap+0lgloM2zX4kftn5B1IpYzTqpyFQ/4Bt0mAxAHeHYNnQxqXmRinvuNg # xVBdJkf77S2uPoCj7GH8BLuxBG5AvftBdsOECS1UkxBvMgEdgkFiDNYiOTx4OtiF # cMSkqTtF2hfQz3zQSku2Ws3IfDReb6e3mmdglTcaarps0wjUjsZvkgFkriK9tUKJ # m/s80FiocSk1VYLZlDwFt+cVFBURJg6zMUjZa/zbCclF83bRVFLeGkuAhHiGPMvS # GmhgaTzVyhYn4p0+8y9oHRaQT/aofEnS5xLrfxnGpTXiUOeSLsJygoLPp66bkDX1 # ZlAeSpQl92QOMeRxykvq6gbylsXQskBBBnGy3tW/AMOMCZIVNSaz7BX8VtYGqLt9 # MmeOreGPRdtBx3yGOP+rx3rKWDEJlIqLXvJWnY0v5ydPpOjL6s36czwzsucuoKs7 # Yk/ehb//Wx+5kMqIMRvUBDx6z1ev+7psNOdgJMoiwOrUG2ZdSoQbU2rMkpLiQ6bG # RinZbI4OLu9BMIFm1UUl9VnePs6BaaeEWvjJSjNm2qA+sdFUeEY0qVjPKOWug/G6 # X5uAiynM7Bu2ayBjUwIDAQABo4IBXTCCAVkwEgYDVR0TAQH/BAgwBgEB/wIBADAd # BgNVHQ4EFgQU729TSunkBnx6yuKQVvYv1Ensy04wHwYDVR0jBBgwFoAU7NfjgtJx # XWRM3y5nP+e6mK4cD08wDgYDVR0PAQH/BAQDAgGGMBMGA1UdJQQMMAoGCCsGAQUF # BwMIMHcGCCsGAQUFBwEBBGswaTAkBggrBgEFBQcwAYYYaHR0cDovL29jc3AuZGln # aWNlcnQuY29tMEEGCCsGAQUFBzAChjVodHRwOi8vY2FjZXJ0cy5kaWdpY2VydC5j # b20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNydDBDBgNVHR8EPDA6MDigNqA0hjJo # dHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkUm9vdEc0LmNy # bDAgBgNVHSAEGTAXMAgGBmeBDAEEAjALBglghkgBhv1sBwEwDQYJKoZIhvcNAQEL # BQADggIBABfO+xaAHP4HPRF2cTC9vgvItTSmf83Qh8WIGjB/T8ObXAZz8OjuhUxj # aaFdleMM0lBryPTQM2qEJPe36zwbSI/mS83afsl3YTj+IQhQE7jU/kXjjytJgnn0 # hvrV6hqWGd3rLAUt6vJy9lMDPjTLxLgXf9r5nWMQwr8Myb9rEVKChHyfpzee5kH0 # F8HABBgr0UdqirZ7bowe9Vj2AIMD8liyrukZ2iA/wdG2th9y1IsA0QF8dTXqvcnT # mpfeQh35k5zOCPmSNq1UH410ANVko43+Cdmu4y81hjajV/gxdEkMx1NKU4uHQcKf # ZxAvBAKqMVuqte69M9J6A47OvgRaPs+2ykgcGV00TYr2Lr3ty9qIijanrUR3anzE # wlvzZiiyfTPjLbnFRsjsYg39OlV8cipDoq7+qNNjqFzeGxcytL5TTLL4ZaoBdqbh # OhZ3ZRDUphPvSRmMThi0vw9vODRzW6AxnJll38F0cuJG7uEBYTptMSbhdhGQDpOX # gpIUsWTjd6xpR6oaQf/DJbg3s6KCLPAlZ66RzIg9sC+NJpud/v4+7RWsWCiKi9EO # LLHfMR2ZyJ/+xhCx9yHbxtl5TPau1j/1MIDpMPx0LckTetiSuEtQvLsNz3Qbp7wG # WqbIiOWCnb5WqxL3/BAPvIXKUjPSxyZsq8WhbaM2tszWkPZPubdcMIIG7TCCBNWg # AwIBAgIQCoDvGEuN8QWC0cR2p5V0aDANBgkqhkiG9w0BAQsFADBpMQswCQYDVQQG # EwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xQTA/BgNVBAMTOERpZ2lDZXJ0 # IFRydXN0ZWQgRzQgVGltZVN0YW1waW5nIFJTQTQwOTYgU0hBMjU2IDIwMjUgQ0Ex # MB4XDTI1MDYwNDAwMDAwMFoXDTM2MDkwMzIzNTk1OVowYzELMAkGA1UEBhMCVVMx # FzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMTswOQYDVQQDEzJEaWdpQ2VydCBTSEEy # NTYgUlNBNDA5NiBUaW1lc3RhbXAgUmVzcG9uZGVyIDIwMjUgMTCCAiIwDQYJKoZI # hvcNAQEBBQADggIPADCCAgoCggIBANBGrC0Sxp7Q6q5gVrMrV7pvUf+GcAoB38o3 # zBlCMGMyqJnfFNZx+wvA69HFTBdwbHwBSOeLpvPnZ8ZN+vo8dE2/pPvOx/Vj8Tch # TySA2R4QKpVD7dvNZh6wW2R6kSu9RJt/4QhguSssp3qome7MrxVyfQO9sMx6ZAWj # FDYOzDi8SOhPUWlLnh00Cll8pjrUcCV3K3E0zz09ldQ//nBZZREr4h/GI6Dxb2Uo # yrN0ijtUDVHRXdmncOOMA3CoB/iUSROUINDT98oksouTMYFOnHoRh6+86Ltc5zjP # KHW5KqCvpSduSwhwUmotuQhcg9tw2YD3w6ySSSu+3qU8DD+nigNJFmt6LAHvH3KS # uNLoZLc1Hf2JNMVL4Q1OpbybpMe46YceNA0LfNsnqcnpJeItK/DhKbPxTTuGoX7w # JNdoRORVbPR1VVnDuSeHVZlc4seAO+6d2sC26/PQPdP51ho1zBp+xUIZkpSFA8vW # doUoHLWnqWU3dCCyFG1roSrgHjSHlq8xymLnjCbSLZ49kPmk8iyyizNDIXj//cOg # rY7rlRyTlaCCfw7aSUROwnu7zER6EaJ+AliL7ojTdS5PWPsWeupWs7NpChUk555K # 096V1hE0yZIXe+giAwW00aHzrDchIc2bQhpp0IoKRR7YufAkprxMiXAJQ1XCmnCf # gPf8+3mnAgMBAAGjggGVMIIBkTAMBgNVHRMBAf8EAjAAMB0GA1UdDgQWBBTkO/zy # Me39/dfzkXFjGVBDz2GM6DAfBgNVHSMEGDAWgBTvb1NK6eQGfHrK4pBW9i/USezL # TjAOBgNVHQ8BAf8EBAMCB4AwFgYDVR0lAQH/BAwwCgYIKwYBBQUHAwgwgZUGCCsG # AQUFBwEBBIGIMIGFMCQGCCsGAQUFBzABhhhodHRwOi8vb2NzcC5kaWdpY2VydC5j # b20wXQYIKwYBBQUHMAKGUWh0dHA6Ly9jYWNlcnRzLmRpZ2ljZXJ0LmNvbS9EaWdp # Q2VydFRydXN0ZWRHNFRpbWVTdGFtcGluZ1JTQTQwOTZTSEEyNTYyMDI1Q0ExLmNy # dDBfBgNVHR8EWDBWMFSgUqBQhk5odHRwOi8vY3JsMy5kaWdpY2VydC5jb20vRGln # aUNlcnRUcnVzdGVkRzRUaW1lU3RhbXBpbmdSU0E0MDk2U0hBMjU2MjAyNUNBMS5j # cmwwIAYDVR0gBBkwFzAIBgZngQwBBAIwCwYJYIZIAYb9bAcBMA0GCSqGSIb3DQEB # CwUAA4ICAQBlKq3xHCcEua5gQezRCESeY0ByIfjk9iJP2zWLpQq1b4URGnwWBdEZ # D9gBq9fNaNmFj6Eh8/YmRDfxT7C0k8FUFqNh+tshgb4O6Lgjg8K8elC4+oWCqnU/ # ML9lFfim8/9yJmZSe2F8AQ/UdKFOtj7YMTmqPO9mzskgiC3QYIUP2S3HQvHG1FDu # +WUqW4daIqToXFE/JQ/EABgfZXLWU0ziTN6R3ygQBHMUBaB5bdrPbF6MRYs03h4o # bEMnxYOX8VBRKe1uNnzQVTeLni2nHkX/QqvXnNb+YkDFkxUGtMTaiLR9wjxUxu2h # ECZpqyU1d0IbX6Wq8/gVutDojBIFeRlqAcuEVT0cKsb+zJNEsuEB7O7/cuvTQasn # M9AWcIQfVjnzrvwiCZ85EE8LUkqRhoS3Y50OHgaY7T/lwd6UArb+BOVAkg2oOvol # /DJgddJ35XTxfUlQ+8Hggt8l2Yv7roancJIFcbojBcxlRcGG0LIhp6GvReQGgMgY # xQbV1S3CrWqZzBt1R9xJgKf47CdxVRd/ndUlQ05oxYy2zRWVFjF7mcr4C34Mj3oc # CVccAvlKV9jEnstrniLvUxxVZE/rptb7IRE2lskKPIJgbaP5t2nGj/ULLi49xTcB # ZU8atufk+EMF/cWuiC7POGT75qaL6vdCvHlshtjdNXOCIUjsarfNZzCCB0kwggUx # oAMCAQICEAe0P3SLJmcoVNrErUyxTt0wDQYJKoZIhvcNAQELBQAwaTELMAkGA1UE # BhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMUEwPwYDVQQDEzhEaWdpQ2Vy # dCBUcnVzdGVkIEc0IENvZGUgU2lnbmluZyBSU0E0MDk2IFNIQTM4NCAyMDIxIENB # MTAeFw0yNTEyMzEwMDAwMDBaFw0yOTAxMDIyMzU5NTlaMIHRMRMwEQYLKwYBBAGC # NzwCAQMTAlVTMRkwFwYLKwYBBAGCNzwCAQITCERlbGF3YXJlMR0wGwYDVQQPDBRQ # cml2YXRlIE9yZ2FuaXphdGlvbjEQMA4GA1UEBRMHMzQwNzk4NTELMAkGA1UEBhMC # VVMxETAPBgNVBAgTCElsbGlub2lzMRAwDgYDVQQHEwdDaGljYWdvMR0wGwYDVQQK # ExRLZWVwZXIgU2VjdXJpdHkgSW5jLjEdMBsGA1UEAxMUS2VlcGVyIFNlY3VyaXR5 # IEluYy4wggGiMA0GCSqGSIb3DQEBAQUAA4IBjwAwggGKAoIBgQCUcNMoSVmxAi0a # vG+StFJMNFFTUIOo3HdBZ+0gqA1XpNgUx11vB1vCZrvFsD9m5oA58tdp4gZN3LmQ # aMvCl2ANUT7MilI02Hf1RWlygBzon6iE0GpU3lgRrwrk1dhtLpGsR6dbMKUUHprc # vKpXk90/VN+vhzY1uik1tCTxkDCPu/AYJg7m9+tR2KqvMuYMaMLhii66eWUAGsBC # h/uZxjkGoJF6qZ0DgFd7rW7VYljbfYSNPeZNGTDgB0J/wOsKl0mn612DTseIvAKt # 4vra/FLFukyEyStnfQ8lWYDcLLCMCjNVrzGipmT5E2iyx7Y1RZCIpNwVogp3Ixbk # Gbq5A/41YNOLLd4cFewyB2F037RevBCRsUODZEt1qBf7Jbu3DiYo1G+zTj9E0R1s # FzyijcfdsTm6X5ble+yCJeGkX5XgsyPnZpyz/FX9Fr0N9pMPGWwW2PKyHEnSytXm # 0Dxdq2P4mA4CBUxq7YoV26L2PF6QEh9BQdXTPcnLysUv7SI/a0ECAwEAAaOCAgIw # ggH+MB8GA1UdIwQYMBaAFGg34Ou2O/hfEYb7/mF7CIhl9E5CMB0GA1UdDgQWBBRG # 4H6CH8pvNX632bsdnrda4MtJLDA9BgNVHSAENjA0MDIGBWeBDAEDMCkwJwYIKwYB # BQUHAgEWG2h0dHA6Ly93d3cuZGlnaWNlcnQuY29tL0NQUzAOBgNVHQ8BAf8EBAMC # B4AwEwYDVR0lBAwwCgYIKwYBBQUHAwMwgbUGA1UdHwSBrTCBqjBToFGgT4ZNaHR0 # cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZEc0Q29kZVNpZ25p # bmdSU0E0MDk2U0hBMzg0MjAyMUNBMS5jcmwwU6BRoE+GTWh0dHA6Ly9jcmw0LmRp # Z2ljZXJ0LmNvbS9EaWdpQ2VydFRydXN0ZWRHNENvZGVTaWduaW5nUlNBNDA5NlNI # QTM4NDIwMjFDQTEuY3JsMIGUBggrBgEFBQcBAQSBhzCBhDAkBggrBgEFBQcwAYYY # aHR0cDovL29jc3AuZGlnaWNlcnQuY29tMFwGCCsGAQUFBzAChlBodHRwOi8vY2Fj # ZXJ0cy5kaWdpY2VydC5jb20vRGlnaUNlcnRUcnVzdGVkRzRDb2RlU2lnbmluZ1JT # QTQwOTZTSEEzODQyMDIxQ0ExLmNydDAJBgNVHRMEAjAAMA0GCSqGSIb3DQEBCwUA # A4ICAQA1Wlq0WzJa3N6DgjgBU7nagIJBab1prPARXZreX1MOv9VjnS5o0CrfQLr6 # z3bmWHw7xT8dt6bcSwRixqvPJtv4q8Rvo80O3eUMvMxQzqmi7z1zf+HG+/3G4F+2 # IYegvPc8Ui151XCV9rjA8tvFWRLRMX0ZRxY1zfT027HMw0iYL20z44+Cky//FAnL # iRwoNDGiRkZiHbB9YOftPAYNMG3gm1z3zOW5RdfKPrqvMuijE+dfyLIAA6Immpzu # FMH+Wgn8NnSlot9b4YKycaqqdjd7wXDjPub/oQ7VShuCSBWj+UNOTVh0vcZGackc # H1DLVgwp2dcKlxJiQKtkHT/T6LloY6LTe6+8wkVkr8EAv1W+q/+M1a4Ao+ykFbIA # 2LBEmA9qdgoLtenAYIiEg+48SjMPgyBbVPE3bhL1vIqjEIxYCfdmi6wx33oYX7HB # +bJ7zitHw4GgtpfPV8y8QRZImKmeDOKyXjQPDmQM/Eglm/Ns0GzBkVXM8h6UI34b # WZrHz9sbLSE20m5Svmxftvw5zju+I3WsmS/stNfWlOkwU0niUgwPHaz21kjXEA5A # g+aqv26wodqZcnGOlChoWDvSJ8KKgdOFbeAYKAMp1NY7iWV315zpGH19RipCR1NH # 0ND8iIubk3WGNf2rzEfqlOi3h2ywqVkU6AKXHdO5JV4otSKKEDGCBdkwggXVAgEB # MH0waTELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMUEwPwYD # VQQDEzhEaWdpQ2VydCBUcnVzdGVkIEc0IENvZGUgU2lnbmluZyBSU0E0MDk2IFNI # QTM4NCAyMDIxIENBMQIQB7Q/dIsmZyhU2sStTLFO3TANBglghkgBZQMEAgEFAKCB # hDAYBgorBgEEAYI3AgEMMQowCKACgAChAoAAMBkGCSqGSIb3DQEJAzEMBgorBgEE # AYI3AgEEMBwGCisGAQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMC8GCSqGSIb3DQEJ # BDEiBCB4rfjN+GHHqRbTHn8SAqVHaCB4AlMhPgu/7x7KkDugCDANBgkqhkiG9w0B # AQEFAASCAYAFjP7T8Zb3zB4dL2bDeIlfnKilioC5P+jqVJmwLwkKuKIcZ20qyOHA # dT4bOiTMufsCKerzZNg3pyM5MVDmha1I2or8SGkJPFG0shF9uNbxUY0jmypcs5Jl # Ljt182cu9aVGIQunFQYZO3GhOjbLGMqb1wjW2xo1AD5XrNnndF+HO6NUk94TimA0 # 5lIxglKzu6JSRgYZd5dtWoFUJYBpzw52ZAi8IEbZbt6WFOwaECuRruqnbwyGXuIw # 1VssAZa7MV5kUPl+XzDHv2LKJ8HIy/lmfzKanO6w1D4JeFfEaFJdQTOjZr9WjA+i # xUt7b+cTMvcfeX4sEc2BXs5pXwOY9/EPtUk3NvH7nB6kRhRkmg6Cm2x268oIIsXh # 9KtKPRiFj82e2zowxB80aFN34JgTdUZ+WSWw8jdwQX5y0XUgfEMfM5QUAZyAH1uq # /zo4EXc3eN8hXkQaJhqVXx0lVHueF1vw6EyIkFq2+PZdLI2MUj60uWRr8zAgLRGn # O0BtK9X4GWGhggMmMIIDIgYJKoZIhvcNAQkGMYIDEzCCAw8CAQEwfTBpMQswCQYD # VQQGEwJVUzEXMBUGA1UEChMORGlnaUNlcnQsIEluYy4xQTA/BgNVBAMTOERpZ2lD # ZXJ0IFRydXN0ZWQgRzQgVGltZVN0YW1waW5nIFJTQTQwOTYgU0hBMjU2IDIwMjUg # Q0ExAhAKgO8YS43xBYLRxHanlXRoMA0GCWCGSAFlAwQCAQUAoGkwGAYJKoZIhvcN # AQkDMQsGCSqGSIb3DQEHATAcBgkqhkiG9w0BCQUxDxcNMjYwNTI2MDQwNjQ5WjAv # BgkqhkiG9w0BCQQxIgQgblWmf2Y9pZhKiObtSkHaQvj1BqQg7DDQnclP/JZfxhIw # DQYJKoZIhvcNAQEBBQAEggIAyWV52QTgJS4zgF92ixgtIICP2V7IYCgYZfLE0Kd2 # wh91lYk/ZA2SnN0JCMDeChRE1ueqPUVcIKpi4u8bShfhquO+36eNALlylwEbIkcO # sX4JnyKd+CF5IUOyNt3XxJ58V1uiuzz37i2C6W0+a7guG/VvmTZI+jFXQXwbQIkC # Vf6W3h5F/IePmwCmRY1f7QbWVEg7Ey3/J+cRyNDIPjfqaZo9qWpnfnyqjrke+ANq # Do80VgIEXvqB9nNvGWgiQ27YuaMSdM+UZAdFkl5NJBqRlr0YYt4UPuCfH+VreUIL # mymKyiv88QSNvLxfuTsYYR0dw5+uAhWjglyBnj3fwh04fTIWmVqzLkLKElgtUX3m # YTleysudiqZZ0dWeEJbVFv2lvWVMaYk2HkL/lIxGIZWLiwJtFI+rasoV5pVuVuUS # ZBgDzZnlR0BOVnHhUBHR5ZxiXkUU2vPC0RojpCwMVQcaqxdfZ9uzZqCjOdfkC27T # fCC9cYm9Ds3oOr9eEKv7pEKrVOxzysJGx0cZ/ZLZwNBapyMbR86sT59GCMDUM1BL # dTUFAtMsajEApT8t4ykHCtCBDzg9Us+eJgXc1OQPhDzn9eUQnDcxzrtlgr814DN7 # 6e0r9QmLTi7S5v0ZsBI6497IWKrNel9k8F9AorX9PxBBp4kL49m+TW23ire3JVMU # eWw= # SIG # End signature block |