DEV Community

Cover image for Get the Count of Azure Resources using PowerShell Runbook & Send it to an Email.
Kinza Kanwal
Kinza Kanwal

Posted on

1

Get the Count of Azure Resources using PowerShell Runbook & Send it to an Email.

Background Knowledge 🤩

Before going into the solution, Let's understand a bit about the Kusto Query Language as this is going to be used to fetch the count across subscriptions🙂. Kusto query language is the main key this is used to extract the count from azure portal. Following query is used to give the count of all resources in Azure. You can call it the punch line of this article 😎. Further more You verify this query in an Azure service called Azure Resource Graph Explorer

summarize Count=count() by type
Enter fullscreen mode Exit fullscreen mode

Steps to Configure the Email Server 🔥

  1. 🚀 Get SMTP email service.

  2. 🚀 Add email credentials in Azure Portal > Resources > Credentials tab of side bar.

  3. 🚀 Import modules names Az_ResourceGraph and all the modules that are it's dependency.

Steps to Create Runbook 🔥

1.🚀 Create an automation account.

2.🚀 Create a PowerShell workflow type runbook.

3.🚀 Add the following script in the workflow runbook file.

#------------------- LOGIN TO AZURE -------------------#
try
   {
        "Logging in to Azure..."
        Connect-AzAccount -Identity
   }
   catch 
   {
        Write-Error -Message $_.Exception
        throw $_.Exception
   }
Enter fullscreen mode Exit fullscreen mode

4.🚀 Add Email Creds in the runbook

#------------------- CONFIGURE EMAIL SERVER -------------------#

$Username ="username"
$Password = ConvertTo-SecureString "PasswordString" -AsPlainText -Force
$credential = New-Object System.Management.Automation.PSCredential $Username, $Password
$SMTPServer = "smtp.sendgrid.net"
$EmailFrom = "youremail.com"  

[string[]]$EmailTo = "youremail.com"
$Subject = "Attached - Resource Count"


Enter fullscreen mode Exit fullscreen mode

5.🚀 Create variables to store the count in.

#------------------- CALCULATING COUNT -------------------#
$WebCount = 0
$DbCount = 0
$VmCount = 0
Enter fullscreen mode Exit fullscreen mode

6.🚀 Create an array of subscriptions to loop through and get the count of resource thought out the tenant that might contain more than 1 subscription

$ArrayOfSubscriptions  = "Subscription1 ID","Subscription2 ID" , "Subscription3 ID","Subscription4 ID","Subscription5 ID"
Enter fullscreen mode Exit fullscreen mode

7.🚀 Start foreach loop

# START OF LOOP
foreach ($element in $ArrayOfSubscriptions) {
echo $element
Set-AzContext -Subscription $element

class AzResourceGraphException : Exception {
    [string] $additionalData

    AzResourceGraphException($Message, $additionalData) : base($Message) {
        $this.additionalData = $additionalData
            }
        }
Enter fullscreen mode Exit fullscreen mode

8.🚀 Place the following code in a for loop to loop though all the subscription, If there is only one subscription no need to add loop.

  • 📌 Websites📌
#------------------- WEBSITES -------------------#
try {
    $resourceGraphQuery = "Resources | where type =~ 'Microsoft.Web/sites' | summarize count()" 

    $WebCount = Search-AzGraph -Subscription $element -Query $resourceGraphQuery -ErrorVariable grapherror -ErrorAction SilentlyContinue 

    if ($null -ne $grapherror.Length) {

    $errorJSON = $grapherror.ErrorDetails.Message | ConvertFrom-Json

    throw [AzResourceGraphException]::new($errorJSON.error.details.code, $errorJSON.error.details.message)

            }
        }
catch [AzResourceGraphException] {
            Write-Host "An error on KQL query"
            Write-Host $_.Exception.message
            Write-Host $_.Exception.additionalData
        }
catch {
    Write-Host "An error occurred in the script"
    Write-Host $_.Exception.message
        }      
Enter fullscreen mode Exit fullscreen mode

-📌 Databases📌

#------------------- DATABASE -------------------#

try {
     $resourceGraphQuery = "Resources | where type =~ 'Microsoft.Sql/Servers/databases' | summarize count()" 

     $DbCount = Search-AzGraph -Subscription $element -Query $resourceGraphQuery -ErrorVariable grapherror -ErrorAction SilentlyContinue 

     if ($null -ne $grapherror.Length) {

     $errorJSON = $grapherror.ErrorDetails.Message | ConvertFrom-Json

        throw [AzResourceGraphException]::new($errorJSON.error.details.code, $errorJSON.error.details.message)

            }
        }
        catch [AzResourceGraphException] {
            Write-Host "An error on KQL query"
            Write-Host $_.Exception.message
            Write-Host $_.Exception.additionalData
        }
        catch {
            Write-Host "An error occurred in the script"
            Write-Host $_.Exception.message
        }
Enter fullscreen mode Exit fullscreen mode

-📌 Virtual Machines📌

#------------------- VIRTUAL MACHINES -------------------#
try {
    $resourceGraphQuery = "Resources | where type =~ 'Microsoft.Compute/virtualMachines' | summarize count()" 

    $VmCount = Search-AzGraph -Subscription $element -Query $resourceGraphQuery -ErrorVariable grapherror -ErrorAction SilentlyContinue 

    if ($null -ne $grapherror.Length) {

    $errorJSON = $grapherror.ErrorDetails.Message | ConvertFrom-Json

    throw [AzResourceGraphException]::new($errorJSON.error.details.code, $errorJSON.error.details.message)

        }
    }
catch [AzResourceGraphException] {
    Write-Host "An error on KQL query"
    Write-Host $_.Exception.message
    Write-Host $_.Exception.additionalData
}
catch {
    Write-Host "An error occurred in the script"
    Write-Host $_.Exception.message
}
# END OF LOOP
Enter fullscreen mode Exit fullscreen mode

9.🚀 Store the values in variables

# ECHO RESULT IS DISPLAYED IN OUTPUT WINDOW OF YOUR RUNBOOK

echo "Web Count" $WebCount.count_
echo "Db Count" $DbCount.count_
echo "VM Count" $VmCount.count_
Enter fullscreen mode Exit fullscreen mode

10.🚀 Sum Counts

# COMBINE THE COUNTS FOR ALL SUBSCRIPTION TO GET THE COUNT FOR TENANT
$WebsiteCount = $WebsiteCount + $WebCount.count_
$DatabaseCount = $DatabaseCount + $DbCount.count_
$VirtualMachineCount = $VirtualMachineCount + $VmCount.count_
Enter fullscreen mode Exit fullscreen mode

11.🚀 Last step is to send the email.

#------------------- SENDING EMAIL -------------------#

$Body = "Count of Azure Resources = "
Send-MailMessage -smtpServer $SMTPServer -Credential $credential -Usessl -Port 587  -from $EmailFrom -to $EmailTo -subject $Subject -Body "Count of Tenant Resources:  <br/><br/><br/>Count of Azure Websites : $WebsiteCount <br/><br/><br/> Count of Azure Databases : $DatabaseCount <br/><br/><br/> Count of Azure VirtualMachines : $VirtualMachineCount " -BodyAsHtml
Enter fullscreen mode Exit fullscreen mode

The Complete Solution😍

❤️For complete solution refer to the following gist.

Thank You so much for reading, stay tunned for the future posts.😍❤️

Imagine monitoring actually built for developers

Billboard image

Join Vercel, CrowdStrike, and thousands of other teams that trust Checkly to streamline monitor creation and configuration with Monitoring as Code.

Start Monitoring

Top comments (0)

Image of AssemblyAI

Automatic Speech Recognition with AssemblyAI

Experience near-human accuracy, low-latency performance, and advanced Speech AI capabilities with AssemblyAI's Speech-to-Text API. Sign up today and get $50 in API credit. No credit card required.

Try the API

👋 Kindness is contagious

Dive into an ocean of knowledge with this thought-provoking post, revered deeply within the supportive DEV Community. Developers of all levels are welcome to join and enhance our collective intelligence.

Saying a simple "thank you" can brighten someone's day. Share your gratitude in the comments below!

On DEV, sharing ideas eases our path and fortifies our community connections. Found this helpful? Sending a quick thanks to the author can be profoundly valued.

Okay