mirror of
https://github.com/jbowdre/runtimeterror.git
synced 2025-02-20 01:22:14 +00:00
244 lines
13 KiB
Markdown
244 lines
13 KiB
Markdown
|
---
|
||
|
series: vRA8
|
||
|
date: "2021-03-29T08:34:30Z"
|
||
|
thumbnail: VZaK4btzl.png
|
||
|
usePageBundles: true
|
||
|
tags:
|
||
|
- vmware
|
||
|
- vra
|
||
|
title: 'vRA8 Custom Provisioning: Part One'
|
||
|
---
|
||
|
|
||
|
I recently shared [some details about my little self-contained VMware homelab](/vmware-home-lab-on-intel-nuc-9) as well as how I [integrated {php}IPAM into vRealize Automation 8 for assigning IPs to deployed VMs](/integrating-phpipam-with-vrealize-automation-8). For my next trick, I'll be crafting a flexible Cloud Template and accompanying vRealize Orchestrator workflow that will help to deploy and configure virtual machines based on a vRA user's input. Buckle up, this is going to be A Ride.
|
||
|
|
||
|
### Objectives
|
||
|
Before getting into the *how* it would be good to start with the *what* - what exactly are we hoping to accomplish here? For my use case, I'll need a solution which can:
|
||
|
- use a single Cloud Template to provision Windows VMs to one of several designated compute resources across multiple sites.
|
||
|
- generate a unique VM name which matches a defined naming standard.
|
||
|
- allow the requester to specify which site-specific network should be used, and leverage {php}IPAM to assign a static IP.
|
||
|
- pre-stage a computer account in Active Directory in a site-specific Organizational Unit and automatically join the new computer to the domain.
|
||
|
- create a static record in Microsoft DNS for non-domain systems.
|
||
|
- expand the VM's virtual disk *and extend the volume inside the guest* based on request input.
|
||
|
- add specified domain accounts to the guest's local Administrators group based on request input.
|
||
|
- annotate the VM in vCenter with a note to describe the server's purpose and custom attributes to identify the responsible party and originating ticket number.
|
||
|
|
||
|
Looking back, that's kind of a lot. I can see why I've been working on this for months!
|
||
|
|
||
|
### vSphere setup
|
||
|
In production, I'll want to be able to deploy to different computer clusters spanning multiple vCenters. That's a bit difficult to do on a single physical server, but I still wanted to be able to simulate that sort of dynamic resource selection. So for development and testing in my lab, I'll be using two sites - `BOW` and `DRE`. I ditched the complicated "just because I can" vSAN I'd built previously and instead spun up two single-host nested clusters, one for each of my sites:
|
||
|
data:image/s3,"s3://crabby-images/e10c7/e10c799fdd7365c4310952b7420c7cc1dc02ad12" alt="vCenter showing the BOW and DRE clusters"
|
||
|
|
||
|
Those hosts have one virtual NIC each on a standard switch connected to my home network, and a second NIC each connected to the ["isolated" internal lab network](vmware-home-lab-on-intel-nuc-9#networking) with all the VLANs for the guests to run on:
|
||
|
data:image/s3,"s3://crabby-images/0671a/0671a2efab77eed0f8ceb980225b4c7ee7126e34" alt="dvSwitch showing attached hosts and dvPortGroups"
|
||
|
|
||
|
### vRA setup
|
||
|
On the vRA side of things, I logged in to the Cloud Assembly portion and went to the Infrastructure tab. I first created a Project named `LAB`, added the vCenter as a Cloud Account, and then created a Cloud Zone for the vCenter as well. On the Compute tab of the Cloud Zone properties, I manually added both the `BOW` and `DRE` clusters.
|
||
|
data:image/s3,"s3://crabby-images/18452/18452eeca84e3a7ffee152bf218a228b25649947" alt="BOW and DRE Clusters added to Cloud Zone"
|
||
|
|
||
|
I also created a Network Profile and added each of the nested dvPortGroups I had created for this purpose.
|
||
|
data:image/s3,"s3://crabby-images/96fb9/96fb9c122adfc7474a2409c8750f418d2949ea4a" alt="Network Profile with added vSphere networks"
|
||
|
|
||
|
Each network also gets associated with the related IP Range which was [imported from {php}IPAM](/integrating-phpipam-with-vrealize-automation-8).
|
||
|
data:image/s3,"s3://crabby-images/fbbd2/fbbd2f668a72e45d8210818a2d485200926c0814" alt="IP Range bound to a network"
|
||
|
|
||
|
Since each of my hosts only has 100GB of datastore and my Windows template specifies a 60GB VMDK, I went ahead and created a Storage Profile so that deployments would default to being Thin Provisioned.
|
||
|
data:image/s3,"s3://crabby-images/d9c38/d9c38ae69becd3172a70058cabca3027ef6b6200" alt="Thin-provision storage profile"
|
||
|
|
||
|
I created a few Flavor Mappings ranging from `micro` (1vCPU|1GB RAM) to `giant` (8vCPU|16GB) but for this resource-constrained lab I'll stick mostly to the `micro`, `tiny` (1vCPU|2GB), and `small` (2vCPU|2GB) sizes.
|
||
|
data:image/s3,"s3://crabby-images/ae057/ae05760d6cd0af77446de1f1c31a2627ac2fe65b" alt="T-shirt size Flavor Mappings"
|
||
|
|
||
|
And I created an Image Mapping named `ws2019` which points to a Windows Server 2019 Core template I have stored in my lab's Content Library (cleverly-named "LABrary" for my own amusement).
|
||
|
data:image/s3,"s3://crabby-images/3742a/3742a4188135575cb541b71a6d5737daadcc2066" alt="Windows Server Image Mapping"
|
||
|
|
||
|
And with that, my vRA infrastructure is ready for testing a *very* basic deployment.
|
||
|
|
||
|
### My First Cloud Template
|
||
|
Now it's time to leave the Infrastructure tab and visit the Design one, where I'll create a new Cloud Template (what previous versions of vRA called "Blueprints"). I start by dragging one each of the **vSphere > Machine** and **vSphere > Network** entities onto the workspace. I then pop over to the Code tab on the right to throw together some simple YAML statements:
|
||
|
data:image/s3,"s3://crabby-images/0e3bf/0e3bfdb2b5a5fdd7867b53305c3b5ecaab30bacc" alt="My first Cloud Template!"
|
||
|
|
||
|
VMware's got a [pretty great document](https://docs.vmware.com/en/vRealize-Automation/8.3/Using-and-Managing-Cloud-Assembly/GUID-6BA1DA96-5C20-44BF-9C81-F8132B9B4872.html#list-of-input-properties-2) describing the syntax for these input properties, plus a lot of it is kind of self-explanatory. Let's step through this real quick:
|
||
|
```yaml
|
||
|
formatVersion: 1
|
||
|
inputs:
|
||
|
# Image Mapping
|
||
|
image:
|
||
|
type: string
|
||
|
title: Operating System
|
||
|
oneOf:
|
||
|
- title: Windows Server 2019
|
||
|
const: ws2019
|
||
|
default: ws2019
|
||
|
```
|
||
|
`formatVersion` is always gonna be 1 so we'll skip right past that.
|
||
|
|
||
|
The first input is going to ask the user to select the desired Operating System for this deployment. The `oneOf` type will be presented as a dropdown (with only one option in this case, but I'll leave it this way for future flexibility); the user will see the friendly "Windows Server 2019" `title` which is tied to the `ws2019` `const` value. For now, I'll also set the `default` value of the field so I don't have to actually click the dropdown each time I test the deployment.
|
||
|
|
||
|
```yaml
|
||
|
# Flavor Mapping
|
||
|
size:
|
||
|
title: Resource Size
|
||
|
type: string
|
||
|
oneOf:
|
||
|
- title: 'Micro [1vCPU|1GB]'
|
||
|
const: micro
|
||
|
- title: 'Tiny [1vCPU|2GB]'
|
||
|
const: tiny
|
||
|
- title: 'Small [2vCPU|2GB]'
|
||
|
const: small
|
||
|
default: small
|
||
|
```
|
||
|
|
||
|
Now I'm asking the user to pick the t-shirt size of the VM. These will correspond to the Flavor Mappings I defined earlier. I again chose to use a `oneOf` data type so that I can show the user more information for each option than is embedded in the name. And I'm setting a `default` value to avoid unnecessary clicking.
|
||
|
|
||
|
The `resources` section is where the data from the inputs gets applied to the deployment:
|
||
|
|
||
|
```yaml
|
||
|
resources:
|
||
|
Cloud_vSphere_Machine_1:
|
||
|
type: Cloud.vSphere.Machine
|
||
|
properties:
|
||
|
image: '${input.image}'
|
||
|
flavor: '${input.size}'
|
||
|
networks:
|
||
|
- network: '${resource.Cloud_vSphere_Network_1.id}'
|
||
|
assignment: static
|
||
|
Cloud_vSphere_Network_1:
|
||
|
type: Cloud.vSphere.Network
|
||
|
properties:
|
||
|
networkType: existing
|
||
|
```
|
||
|
|
||
|
So I'm connecting the selected `input.image` to the Image Mapping configured in vRA, and the selected `input.size` goes back to the Flavor Mapping that will be used for the deployment. I also specify that `Cloud_vSphere_Machine_1` should be connected to `Cloud_vSphere_Network_1` and that it should use a `static` (as opposed to `dynamic`) IP address. Finally, vRA is told that the `Cloud_vSphere_Network_1` should be an existing vSphere network.
|
||
|
|
||
|
All together now:
|
||
|
|
||
|
```yaml
|
||
|
formatVersion: 1
|
||
|
inputs:
|
||
|
# Image Mapping
|
||
|
image:
|
||
|
type: string
|
||
|
title: Operating System
|
||
|
oneOf:
|
||
|
- title: Windows Server 2019
|
||
|
const: ws2019
|
||
|
default: ws2019
|
||
|
# Flavor Mapping
|
||
|
size:
|
||
|
title: Resource Size
|
||
|
type: string
|
||
|
oneOf:
|
||
|
- title: 'Micro [1vCPU|1GB]'
|
||
|
const: micro
|
||
|
- title: 'Tiny [1vCPU|2GB]'
|
||
|
const: tiny
|
||
|
- title: 'Small [2vCPU|2GB]'
|
||
|
const: small
|
||
|
default: small
|
||
|
resources:
|
||
|
Cloud_vSphere_Machine_1:
|
||
|
type: Cloud.vSphere.Machine
|
||
|
properties:
|
||
|
image: '${input.image}'
|
||
|
flavor: '${input.size}'
|
||
|
networks:
|
||
|
- network: '${resource.Cloud_vSphere_Network_1.id}'
|
||
|
assignment: static
|
||
|
Cloud_vSphere_Network_1:
|
||
|
type: Cloud.vSphere.Network
|
||
|
properties:
|
||
|
networkType: existing
|
||
|
```
|
||
|
|
||
|
Cool! But does it work? Hitting the **Test** button at the bottom right is a great way to validate a template before actually running a deployment. That will confirm that the template syntax, infrastructure, and IPAM configuration is all set up correctly to support this particular deployment.
|
||
|
data:image/s3,"s3://crabby-images/fc614/fc6146e52a89c13a9b17dfc92a10323fe449cb98" alt="Test inputs"
|
||
|
data:image/s3,"s3://crabby-images/7c8c4/7c8c471144a9cc806d1693dff1c7d47926b605e8" alt="Test results"
|
||
|
|
||
|
Looks good! I like to click on the **Provisioning Diagram** link to see a bit more detail about where components were placed and why. That's also an immensely helpful troubleshooting option if the test *isn't* successful.
|
||
|
data:image/s3,"s3://crabby-images/8961c/8961cd8ffc0456ef51f2b26e136eedf013686215" alt="Provisioning diagram"
|
||
|
|
||
|
And finally, I can hit that **Deploy** button to actually spin up this VM.
|
||
|
data:image/s3,"s3://crabby-images/2f762/2f76268a464261e5046f68596f495e7b8b625b17" alt="Deploy this sucker"
|
||
|
|
||
|
Each deployment has to have a *unique* deployment name. I got tired of trying to keep up with what names I had already used so kind of settled on a [DATE]_[TIME] format for my test deployments. I'll automatic this tedious step away in the future.
|
||
|
|
||
|
I then confirm that the (automatically-selected default) inputs are correct and kick it off.
|
||
|
data:image/s3,"s3://crabby-images/85783/85783754b62831b9935022c866a281f8a215ff4b" alt="Deployment inputs"
|
||
|
|
||
|
The deployment will take a few minutes. I like to click over to the **History** tab to see a bit more detail as things progress.
|
||
|
data:image/s3,"s3://crabby-images/94abc/94abcb5b942595ffad52e4bad5cf66e2cb1e6bd6" alt="Deployment history"
|
||
|
|
||
|
It doesn't take too long for activity to show up on the vSphere side of things:
|
||
|
data:image/s3,"s3://crabby-images/b568e/b568e4ec855478d4b914c4410000dd6d1aeb142c" alt="vSphere is cloning the source template"
|
||
|
|
||
|
And there's the completed VM - notice the statically-applied IP address courtesy of {php}IPAM!
|
||
|
data:image/s3,"s3://crabby-images/908bf/908bf9634ddf97eaa2f3fab79613c7cf7ff538bb" alt="Completed test VM"
|
||
|
|
||
|
And I can pop over to the IPAM interface to confirm that the IP has been marked as reserved as well:
|
||
|
data:image/s3,"s3://crabby-images/68bf8/68bf814e17d8108c96a3cdb45582a1eed4d2843c" alt="Newly-created IPAM reservation"
|
||
|
|
||
|
Fantastic! But one of my objectives from earlier was to let the user control where a VM gets provisioned. Fortunately it's pretty easy to implement thanks to vRA 8's use of tags.
|
||
|
|
||
|
### Using tags for resource placement
|
||
|
Just about every entity within vRA 8 can have tags applied to it, and you can leverage those tags in some pretty creative and useful ways. For now, I'll start by applying tags to my compute resources; I'll use `comp:bow` for the "BOW Cluster" and `comp:dre` for the "DRE Cluster".
|
||
|
data:image/s3,"s3://crabby-images/53142/53142957927bee124befcc5bf5cd7ddd9f4d14b6" alt="Compute tags"
|
||
|
|
||
|
I'll also use the `net:bow` and `net:dre` tags to logically divide up the networks between my sites:
|
||
|
data:image/s3,"s3://crabby-images/cfa3c/cfa3c56677549bc0d53cc8798c96163769c603bf" alt="Network tags"
|
||
|
|
||
|
I can now add an input to the Cloud Template so the user can pick which site they need to deploy to:
|
||
|
|
||
|
```yaml
|
||
|
inputs:
|
||
|
# Datacenter location
|
||
|
site:
|
||
|
type: string
|
||
|
title: Site
|
||
|
enum:
|
||
|
- BOW
|
||
|
- DRE
|
||
|
# Image Mapping
|
||
|
```
|
||
|
|
||
|
I'm using the `enum` option now instead of `oneOf` since the site names shouldn't require further explanation.
|
||
|
|
||
|
And then I'll add some `constraints` to the `resources` section, making use of the `to_lower` function from the [cloud template expression syntax](https://docs.vmware.com/en/vRealize-Automation/8.3/Using-and-Managing-Cloud-Assembly/GUID-12F0BC64-6391-4E5F-AA48-C5959024F3EB.html) to automatically convert the selected site name from all-caps to lowercase so it matches the appropriate tag:
|
||
|
|
||
|
```yaml
|
||
|
resources:
|
||
|
Cloud_vSphere_Machine_1:
|
||
|
type: Cloud.vSphere.Machine
|
||
|
properties:
|
||
|
image: '${input.image}'
|
||
|
flavor: '${input.size}'
|
||
|
networks:
|
||
|
- network: '${resource.Cloud_vSphere_Network_1.id}'
|
||
|
assignment: static
|
||
|
constraints:
|
||
|
- tag: 'comp:${to_lower(input.site)}'
|
||
|
Cloud_vSphere_Network_1:
|
||
|
type: Cloud.vSphere.Network
|
||
|
properties:
|
||
|
networkType: existing
|
||
|
constraints:
|
||
|
- tag: 'net:${to_lower(input.site)}'
|
||
|
```
|
||
|
|
||
|
So the VM will now only be deployed to the compute resource and networks which are tagged to match the selected Site identifier. I ran another test to make sure I didn't break anything:
|
||
|
data:image/s3,"s3://crabby-images/0a3b3/0a3b359fa06784e51eb7430b65501c9e516d5eb5" alt="Testing against the DRE site"
|
||
|
|
||
|
It came back successful, so I clicked through to see the provisioning diagram. On the network tab, I see that only the last two networks (`d1650-Servers-4` and `d1660-Servers-5`) were considered since the first three didn't match the required `net:dre` tag:
|
||
|
data:image/s3,"s3://crabby-images/a33e1/a33e1a4d073894fdafe97738514da9f2a322d38d" alt="Network provisioning diagram"
|
||
|
|
||
|
And it's a similar story on the compute tab:
|
||
|
data:image/s3,"s3://crabby-images/607b4/607b4fc16681301a0734398b588f13808b5c5afd" alt="Compute provisioning diagram"
|
||
|
|
||
|
As a final test for this change, I kicked off one deployment to each site to make sure things worked as expected.
|
||
|
data:image/s3,"s3://crabby-images/d87d9/d87d915705e6da8786ed1bf819f9c1e51d35b52b" alt="vSphere showing one VM at each site"
|
||
|
|
||
|
Nice!
|
||
|
|
||
|
### Conclusion
|
||
|
This was kind of an easy introduction into what I've been doing with vRA 8 these past several months. The basic infrastructure (both in vSphere and vRA) will support far more interesting and flexible deployments as I dig deeper. For now, being able to leverage vRA tags for placing workloads on specific compute resources is a great start.
|
||
|
|
||
|
Things will get *much* more interesting in the next post, where I'll dig into how I'm using vRealize Orchestrator to generate unique computer names which fit a defined naming standard.
|