Bit of giving something back..., especially as it includes snippets of code that I've been given advice on...!
We've piloted some code to allow users to respond to Approval Requests by replying to the email with a one word response - 'Approve' or 'Deny' (they're actually allowed approve, approved, deny and denied, but you get the drift!).
The code is by no means finished, and it's still got lots of tweaks, consolidation and simplification etc, but in it's current state it may be useful for anyone who wants to do the same thing as it is quite "chatty" and logical to follow. Basically:-
1. Read all mails in mail Inbox
2. Strips and moves to 'garbage' anything that isn't a direct reply to an approval request (based on subject line string)
3. For each mail that is a response, finds the ticket GUID (parses the content for the HTML link in the original email)
4. Tries to Approve or Deny the ticket by calling the Adaxes API to pass a Approve() or Deny() to the relevant pending approval
5. Shifts the approved, denyed, or unknown (couldn't find a 'decision' in the right place) mails to archive folders
Important thing to note is that it used the "NetCmdets" (http://www.netcmdlets.com/) powershell module to implement IMAP based mailbox management functions. Non-commercial (test) version is free to download, so you can play and it only a few hundred $ to purchase. IMAP is a bit of a vague protocol so if you're anything like us you may find that some servers won't delete 'moved' mails, so a bit of the code is to output before\after counts to make sure everything is where it should be.
$env:PSModulePath = $env:PSModulePath + ";C:\Windows\System32\Modules"
Import-Module NetCmdlets
Import-Module Adaxes
$startupVariables=""
new-variable -force -name startupVariables -value ( Get-Variable | % { $_.Name } )
function Cleanup-Variables
{
Get-Variable | Where-Object { $startupVariables -notcontains $_.Name } | % { Remove-Variable -Name "$($_.Name)" -Force -Scope "global" }
}
function RemoveLineCarriage($object)
{
$result = [System.String] $object;
$result = $result -replace "`t","";
$result = $result -replace "`n","";
$result = $result -replace "`r","";
$result = $result -replace [Environment]::NewLine, "";
$result;
}
$mailserver = "localhost"
$msuser = "adaxes@test.net"
$mspass = "password"
$adaxesserver = "adaxes.test.net"
Try
{
$imapstr = Connect-IMAP -server $mailserver -user $msuser -password $mspass
}
Catch
{
Write-Host
Write-Host "Can't do a damn thing if I can't connect to the mail server"
Cleanup-Variables
Continue
}
Try
{
$newmailarray = @(Get-IMAP -connection $imapstr -folder INBOX)
$approvedMail = @(Get-IMAP -connection $imapstr -folder INBOX.Approved)
$deniedMail = @(Get-IMAP -connection $imapstr -folder INBOX.Denied)
$unknownMail = @(Get-IMAP -connection $imapstr -folder INBOX.Unknown)
$garbageMail = @(Get-IMAP -connection $imapstr -folder INBOX.Garbage)
$errorMail = @(Get-IMAP -connection $imapstr -folder INBOX.Reprocess)
}
Catch
{
Write-Host
Write-Host "Required folder structure not present, make sure 'Approved', 'Denied', 'Unknown', 'Garbage' and 'Reprocess' folders have been created"
Cleanup-Variables
Continue
}
Write-Host
If ($newmailarray.count -ne "0")
{
Write-Host "Number of mails in inbox :" $newmailarray.count
}
Else
{
Write-Host "Number of mails in inbox : 0 - quiting early to go for a smoke"
$imapStrShut = Disconnect-IMAP -connection $imapstr
Write-Host
Write-Host "Closed Server Connection" $imapStrShut
Cleanup-Variables
Continue
}
Write-Host
Write-Host "Parsing new mails"
Write-Host "-----------------"
Write-Host
If ($newmailarray.count -ne $NULL)
{
$potential = @()
$garbage = @()
ForEach ($newmail in $newmailarray)
{
If ($newmail.subject -ne "RE: Active Directory - Approval Request")
{
$garbage += $newmail.ID
Write-Host "** Mail ID" $newmail.ID "Ignored **"
Write-Host "Subj: " $newmail.subject
Write-Host "From: " $newmail.fromemail
$gmovestatus = Move-IMAP -connection $imapstr -folder Inbox -Message $newmail.ID.toString() -Destination Inbox.Garbage
Write-Host "Moved message" $newmail.ID "to folder" $gmovestatus.Destination
set-imap -connection $imapstr -folder Inbox -message $newmail.ID.toString() -expunge
Clear-Variable -name gmovestatus
Write-Host
}
Else
{
$potential += $newmail.ID
Write-Host "** Mail ID" $newmail.ID "Tagged For Processing **"
Write-Host "Subj: " $newmail.subject
Write-Host "From: " $newmail.fromemail
Write-Host
}
}
}
Write-Host "Summary Results"
Write-Host "----------------------------"
Write-Host "Number of mails ignored :" $garbage.count
Write-Host "Number of mails to process :" $potential.count
Write-Host "Mail ID's ignored :" $garbage
Write-Host "Mail ID's to be processed :" $potential
If ($potential -ne $NULL)
{
$trash = [Reflection.Assembly]::LoadWithPartialName("Softerra.Adaxes.Adsi")
$admNS = New-Object("Softerra.Adaxes.Adsi.AdmNamespace")
$admService = $admNS.GetServiceDirectly($adaxesserver)
$containerPath = $admService.Backend.GetConfigurationContainerPath("ApprovalRequests")
$container = $admService.OpenObject($containerPath.ToString(), $NULL, $NULL, 0)
$requests = $container.GetApprovalRequests("ADM_APPROVALSTATE_PENDING")
Write-Host
Write-Host "Approval Tickets"
Write-Host "----------------"
Foreach ($approvalReply in $potential)
{
$emailobject = Get-IMAP -connection $imapstr -view $approvalReply
Write-Host
Write-Host "Mail ID :" $approvalReply
Write-Host "From :" $emailobject.FromEmail
$emailObjectRaw = RemoveLineCarriage($emailobject.Text)
$TGPo = $emailObjectRaw.indexOf("Click here to approve or deny the request<http://adaxes.svr.testsvr.net/testingSelfService/ViewObject.aspx?guid=")
$ticketGUID = $emailobjectRaw.SubString($TGPo + 112,36)
$UGPo = $emailObjectRaw.indexOf("<http://adaxes.svr.testsvr.net/testingSelfService/ViewObject.aspx?guid=")
$userGUID = $emailobjectRaw.SubString($UGPo + 71,36)
$replyPos = $emailObjectRaw.indexOf("Content-Transfer-Encoding: 7bit")
$replyText = $emailobjectRaw.SubString($replyPos + 31,8).toLower()
$decision = "UNDETERMINED"
If ($replyText -like "*approve*")
{
$decision = "APPROVED"
}
If ($replyText -like "*denied*" -or
$replyText -like "*deny*")
{
$decision = "DENIED"
}
Write-Host "Ticket GUID : {$ticketGUID}"
Write-Host "Reply Text :" $replyText
Write-Host "Approval :" $decision
Switch ($decision)
{
"APPROVED" {
$ticketFound = $FALSE
ForEach ($requestID in $requests)
{
$guid = New-Object "System.Guid" (,$requestID)
$guid = $guid.ToString("B")
If ($guid -ne "{$ticketGUID}")
{
Continue
}
Else
{
$ticketFound = $TRUE
$requestPath = "Adaxes://<GUID=$guid>"
$request = $admService.OpenObject($requestPath, $NULL, $NULL, 0)
$request.Approve()
Write-Host "GUID Match :" $guid
Write-Host "State : Processed"
Break
}
}
If ($ticketFound -eq $TRUE)
{
$processedItem = Move-IMAP -connection $imapstr -folder Inbox -Message $approvalreply -Destination Inbox.Approved
}
If ($ticketFound -eq $FALSE)
{
Write-Host "GUID Match : None - No pending approval"
$processedItem = Move-IMAP -connection $imapstr -folder Inbox -Message $approvalreply -Destination Inbox.Reprocess
}
set-imap -connection $imapstr -folder Inbox -message $newmail.ID.toString() -expunge
}
"DENIED" {
$ticketFound = $FALSE
ForEach ($requestID in $requests)
{
$guid = New-Object "System.Guid" (,$requestID)
$guid = $guid.ToString("B")
If ($guid -ne "{$ticketGUID}")
{
Continue
}
Else
{
$ticketFound = $TRUE
$requestPath = "Adaxes://<GUID=$guid>"
$request = $admService.OpenObject($requestPath, $NULL, $NULL, 0)
$request.Deny("Denied via email integration")
Write-Host "GUID Match :" $guid
Write-Host "State : Processed"
Break
}
}
If ($ticketFound -eq $TRUE)
{
$processedItem = Move-IMAP -connection $imapstr -folder Inbox -Message $approvalreply -Destination Inbox.Denied
}
If ($ticketFound -eq $FALSE)
{
Write-Host "State : Reprocessing Error"
$processedItem = Move-IMAP -connection $imapstr -folder Inbox -Message $approvalreply -Destination Inbox.Reprocess
}
set-imap -connection $imapstr -folder Inbox -message $newmail.ID.toString() -expunge
}
"UNDETERMINED" {
$processedItem = Move-IMAP -connection $imapstr -folder Inbox -Message $approvalreply -Destination Inbox.Unknown
set-imap -connection $imapstr -folder Inbox -message $newmail.ID.toString() -expunge
}
}
Write-Host "New Folder :" $processedItem.Destination
}
}
Else
{
Write-Host
Write-Host "No new approvals to process, clearing up"
}
Write-Host
Write-Host "Mail totals before processing"
Write-Host "-----------------------------"
Write-Host "Inbox Mails : " $newmailarray.count
Write-Host "Approved Mails: " $approvedMail.count
Write-Host "Denied Mails : " $deniedMail.count
Write-Host "Unknown Mails : " $unknownMail.count
Write-Host "Reprocessed : " $errorMail.count
Write-Host "Garbage Mails : " $garbageMail.count
$pretotal = $newmailarray.count + $approvedMail.count + $deniedMail.count + $unknownMail.count + $garbageMail.count + $errorMail.count
Write-Host "-----------------------------"
Write-Host "Total : " $pretotal
Write-Host "-----------------------------"
Disconnect-IMAP -connection $imapstr
$imapstr2 = Connect-IMAP -server $mailserver -user $msuser -password $mspass
$newInbox = @(Get-IMAP -connection $imapstr2 -folder INBOX)
$approvedMail = @(Get-IMAP -connection $imapstr2 -folder INBOX.Approved)
$deniedMail = @(Get-IMAP -connection $imapstr2 -folder INBOX.Denied)
$unknownMail = @(Get-IMAP -connection $imapstr2 -folder INBOX.Unknown)
$garbageMail = @(Get-IMAP -connection $imapstr2 -folder INBOX.Garbage)
$errorMail = @(Get-IMAP -connection $imapstr2 -folder INBOX.Reprocess)
Write-Host
Write-Host "Count Verification"
Write-Host "----------------------------"
Write-Host "Inbox Mails : " $newInbox.count
Write-Host "Approved Mails: " $approvedMail.count
Write-Host "Denied Mails : " $deniedMail.count
Write-Host "Unknown Mails : " $unknownMail.count
Write-Host "Reprocessed : " $errorMail.count
Write-Host "Garbage Mails : " $garbageMail.count
$posttotal = $newInbox.count + $approvedMail.count + $deniedMail.count + $unknownMail.count + $garbageMail.count + $errorMail.count
Write-Host "-----------------------------"
Write-Host "Total : " $posttotal
Write-Host "-----------------------------"
Write-Host
If ($pretotal -eq $posttotal)
{
Write-Host "Diff: " ($posttotal - $pretotal) " Verification PASSED"
}
Else
{
Write-Host "Diff: " ($posttotal - $pretotal) " Verification FAILED"
}
Write-Host
Write-Host "Closed Server Connection" $imapStrShut
Cleanup-Variables