Context

The silly idea of OS-tan, the personification of OS as cute manga characters made me think of a sillier idea: show server monitoring as anime girl. Let’s write down the requirements.

Requirements

  • Each server/service should be represented by a unique avatar, like an identicon.

Best Implementation

  • Create several dozen of body part layer template on transparent background, assign for each a palette of allowed colors.
  • Generate a unique hash from a service/server name.
  • From the hash, select layers and color and merge them.

Current Crappy Implementation

  • Generate a unique hash from a service/server name.
  • Feed the hash as a seed to a random number generator.
  • Open a Flash game to create your own anime character, and click (pseudo-)randomly.
  • Do a screenshot.

Part 1: Generate a unique seed from a string

This part will be the sanest part of the final result. We will use .Net to generate a MD5 hash of a string, then truncate it from an int128 to an int32, required to seed Get-Random in PowerShell. MD5 shouldn’t be used as a hash to protect data, but for our purpose, which is to generate scattered numbers from an input of a small size, it’s good enough. Knuth probably would recommend something more robust, but it’s good enough for this first step. Now, the code:

function Get-SeedFromString([String] $String) { 
    $bytes = [System.Security.Cryptography.HashAlgorithm]::Create("MD5").ComputeHash([System.Text.Encoding]::UTF8.GetBytes($String))
    [bitconverter]::ToInt32($bytes,0)
}

Part 2: Feed the hash as a seed to a random number generator.

This part is very easy, we just need to set Get-Random.

$seed = Get-SeedFromString $name
Get-Random -SetSeed $seed

Every times SetSeed is called, it will reset the generator, which will ensure that our process can be reproduced.

Part 4: Do a screenshot.

Wait, why is Part 4 before Part 3? Because while we start getting dirty, it’s not full madness yet, and it’s short. We will write a function doing a screenshot of a rectangular area of the screen, needing the coordinate of the top right corner and the bottom left corner.

function Get-ScreenShot([Int]$TopRightX,[Int]$TopRightY,[Int]$BottomLeftX,[Int]$BottomLeftY, [string]$name) {
    $bounds = [Drawing.Rectangle]::FromLTRB($TopRightX,$TopRightY, $BottomLeftX,$BottomLeftY)
    $bmp = New-Object Drawing.Bitmap $bounds.width, $bounds.height
    $graphics = [Drawing.Graphics]::FromImage($bmp)
    $graphics.CopyFromScreen($bounds.Location, [Drawing.Point]::Empty, $bounds.size)

    $screenCapturePathBase = "$pwd\$name.png"
    if (Test-Path $screenCapturePathBase) {
        Remove-Item $screenCapturePathBase
    }
    $bmp.Save($screenCapturePathBase)
    $graphics.Dispose()
    $bmp.Dispose()
}

A proper screenshot function would check if the file does not exist, and maybe increment the name. For our goal, we don’t need that.

Part 3: Open a Flash game to create your own anime character, and click (pseudo-)randomly.

The madness really starts now. Since I have not an artsy bone in my coder body, I will rely on more talented people. I tested a lot of Anime Avatar Generator games and finally decided to work with one of the latest fron Gen-8. The menus are relatively consistant, as is the color palette. We need to do several thing:

  • Package the swf to open it in a webbrowser controlled by Powershell
  • Click on the “play button” once the swf is loaded
  • Click around a bunch of time
  • Screenshot and quit

Helper function: Click somewhere on the screen

I just used some class copy/pasted from Internet, it create a static method to click somewhere. I think it’s buggy for weird multi screen setup, but we are not trying to do thing in the proper way, I think it’s now clear.

$cSource = @'
using System;
using System.Drawing;
using System.Runtime.InteropServices;
using System.Windows.Forms;
public class Clicker
{
//https://msdn.microsoft.com/en-us/library/windows/desktop/ms646270(v=vs.85).aspx
[StructLayout(LayoutKind.Sequential)]
struct INPUT
{ 
    public int        type; // 0 = INPUT_MOUSE,
                            // 1 = INPUT_KEYBOARD
                            // 2 = INPUT_HARDWARE
    public MOUSEINPUT mi;
}

//https://msdn.microsoft.com/en-us/library/windows/desktop/ms646273(v=vs.85).aspx
[StructLayout(LayoutKind.Sequential)]
struct MOUSEINPUT
{
    public int    dx ;
    public int    dy ;
    public int    mouseData ;
    public int    dwFlags;
    public int    time;
    public IntPtr dwExtraInfo;
}

//This covers most use cases although complex mice may have additional buttons
//There are additional constants you can use for those cases, see the msdn page
const int MOUSEEVENTF_MOVED      = 0x0001 ;
const int MOUSEEVENTF_LEFTDOWN   = 0x0002 ;
const int MOUSEEVENTF_LEFTUP     = 0x0004 ;
const int MOUSEEVENTF_RIGHTDOWN  = 0x0008 ;
const int MOUSEEVENTF_RIGHTUP    = 0x0010 ;
const int MOUSEEVENTF_MIDDLEDOWN = 0x0020 ;
const int MOUSEEVENTF_MIDDLEUP   = 0x0040 ;
const int MOUSEEVENTF_WHEEL      = 0x0080 ;
const int MOUSEEVENTF_XDOWN      = 0x0100 ;
const int MOUSEEVENTF_XUP        = 0x0200 ;
const int MOUSEEVENTF_ABSOLUTE   = 0x8000 ;

const int screen_length = 0x10000 ;

//https://msdn.microsoft.com/en-us/library/windows/desktop/ms646310(v=vs.85).aspx
[System.Runtime.InteropServices.DllImport("user32.dll")]
extern static uint SendInput(uint nInputs, INPUT[] pInputs, int cbSize);

public static void LeftClickAtPoint(int x, int y)
{
    //Move the mouse
    INPUT[] input = new INPUT[3];
    input[0].mi.dx = x*(65535/System.Windows.Forms.Screen.PrimaryScreen.Bounds.Width);
    input[0].mi.dy = y*(65535/System.Windows.Forms.Screen.PrimaryScreen.Bounds.Height);
    input[0].mi.dwFlags = MOUSEEVENTF_MOVED | MOUSEEVENTF_ABSOLUTE;
    //Left mouse button down
    input[1].mi.dwFlags = MOUSEEVENTF_LEFTDOWN;
    //Left mouse button up
    input[2].mi.dwFlags = MOUSEEVENTF_LEFTUP;
    SendInput(3, input, Marshal.SizeOf(input[0]));
}
}
'@
Add-Type -TypeDefinition $cSource -ReferencedAssemblies System.Windows.Forms,System.Drawing

Packaging the Flash game to be open by PowerShell

We will use a Form with a WebBrowser object to manipulate the flash generator. For that, we need a html page wrapper, since we cannot just inline a .SWF file. Quick and dirty generator.html content:

<object>
    <embed src="anime_face_maker_2_by_gen8-d30uny4.swf" width="900" height="650"></embed>
</object>

Clicking around

We are really in the “meat” of the generator. Let’s say we have opened the window, clicked on “play” and that the game is fully loaded. We will simply use the Get-Random function to generate coordinate where to click inside the window. To get more colorful picture, we will click first in the menus/feature selections, then in the palette. The proper way would be to define the clicking zone relative to the window, but since we force the position of the window, we will deal only in absolute (like a Sith).

function Invoke-FeatureGenerator{
    1..200 | % {
        $x = 300..940 | Get-Random
        $y = 120..615 | Get-Random
        [Clicker]::LeftClickAtPoint($x,$y)
        # force pickup color
        $x = 50..277 | Get-Random
        $y = 468..598 | Get-Random
        [Clicker]::LeftClickAtPoint($x,$y)
    }
}

Remember, since we forced a seed just before, we can replay the script and it will click at the same place in the same order. We loop 200 hundred times, because why not. Really, 50 would do too, but the difference of speed is negligible (since most of the is spent waiting for the windows and the SWF to load).

Putting it all together

The main function will create a form, add the webpage, click on “Play”, click (pseudo-)randomly around, do a screenshot, and close the form. The interesting part is the use of timers to have sequential actions. Each timer’ trigger unregisters itself (so it runs only once) and prepare the next one. Since the code is a bit long and full of boiler plate, here is just the overall gist, that can be run directly.

Once you have run the content of the gist once, you will have in the current folder the HTML template and the SWF file. You can then just run the command Generate-Avatar "PowerShell" and check the result in the current directory.

Result for Generate-Avatar "PowerShell"

Going further

I’m on the fence, either I make this script into a proper Module, or I just burn it down to the ground. I may want to also add a function to generate different expression, like a sleeping face (when the server is on maintenance) or anger (when the monitoring is getting errors).

PowerShell allows SysAdmins to write simple tools which can be distributed to lower level support or even users, which in turn give more time to SysAdmins to work on more important business (watching YouTube, trolling #powershell on slack…). But, in some case, it’s possible to totally remove the middle-man and write a script that will run periodically, without any user input. This serie of Post will focus on the differences between writing code for users and for full automation. If you want more information about the best way to write user-friendly code, there are already a lot of ressource on that all around the web.

This first post describes a sample task that notify It admins when AD accounts will expire and how to set it using Powershell. The code may look rough, but it will be refined after each post. Let’s dig in!

SendAccountExpiryReport.ps1
$ExpiryLimitInDays = 14
$AccountExpiryReportMailSettings = @{
    Subject    = "[REPORT] account expiring in $ExpiryLimitInDays days"
    From       = 'IT-robot@company.com'
    To         = 'IT-team@company.com'
    Encoding   = 'utf-8'
    SmtpServer = 'ns1.company.com'
}

$expiryDateMin = (Get-Date).AddDays($ExpiryLimitInDays)
$expiryDateMax = (Get-Date).AddDays($ExpiryLimitInDays+1)

$allLimitedAccounts = Get-ADUser -Filter "AccountExpirationdate -like '*'" -Properties AccountExpirationdate | Where-Object { $_.AccountExpirationdate -gt $expiryDateMin -and $_.AccountExpirationdate -lt $expiryDateMax }

$AccountExpiryReportMailSettings['Body'] = if ($allLimitedAccounts) {
    ($allLimitedAccounts | Select-Object @{n='Long Account';e={$_.samaccountname}},@{n='Account Expiration Date';e={$_.AccountExpirationdate}} | ConvertTo-Html) -join "`r`n"
} else {
    "nothing today."
}

Send-MailMessage @AccountExpiryReportMailSettings -BodyAsHtml

Let’s say the file is under C:\Tasks\SendAccountExpiryReport. Even if it’s not as straightforward as it was in VBS/batch, adding a Scheduled Task to run a PowerShell script is still relatively simple.

Register-SendAccountExpiryReport.ps1
$scriptPath = "C:\Batches\Tasks\SendAccountExpiryReport\SendAccountExpiryReport.ps1"
$ActionArgument = "-noProfile -ExecutionPolicy Bypass -File `"$scriptPath`""
$action = New-ScheduledTaskAction -Execute 'Powershell.exe' -Argument $ActionArgument
$trigger =  New-ScheduledTaskTrigger -DaysInterval 1 -Daily -At (Get-Date)
$settings = New-ScheduledTaskSettingsSet -Compatibility Win7 -Hidden -AllowStartIfOnBatteries
Register-ScheduledTask -Action $action -Trigger $trigger -TaskName "MyTask-SendAccountExpiryReport" -Settings $settings

Compared to a batch, you need to execute powershell.exe and pass the script path as parameter. We also use -noProfile to be sure to “isolate” the script from the environment, it makes for easier integration in the long run. In a similar manner, we bypass the current ExecutionPolicy, so we don’t have to set it session/computer wide.

That’s enough code to start to be dangerous, since we don’t do any test, we don’t have log or run it unmonitored, but it’s also a short way to get value from PowerShell. Once the script is running, it will run until the cold death of the Universe (or until someone shut down the VM where it’s running…).

Next time, we generate documentation for the task… but in the laziest way possible. And we do it soon!