Skip to main content

Windows Server with Active Directory Domain

A Virtual Server example running Windows Server with Active Directory

In this example, a Virtual Server running Windows Server 2022 is deployed to host an Active Directory Domain.

The Virtual Server is deployed with a static, internal-only IP. This example also highlights how to create a domain with appropriate DNS configurations, as well as the attributes needed to join other Virtual Servers in your namespace to your Active Directory domain.

Prerequisites

This example presumes that the user already has an active CoreWeave account and has configured their user credentials such that resources may be deployed to their namespace.

The Windows Servers below are deployed using kubectl. To follow along, ensure that kubectl is installed on your local system.

Example source code

To try out this example yourself, first clone the virtual-server-windows-internal-ip-only.yaml manifest from GitHub.

Create a Primary Domain Controller (PDC)

First, the prepared virtual-server-windows-internal-ip-only.yaml manifest is cloned locally.

Click to expand - virtual-server-windows-internal-ip-only.yaml
apiVersion: virtualservers.coreweave.com/v1alpha1
kind: VirtualServer
metadata:
name: vs-pdc
labels:
app.kubernetes.io/component: dc
spec:
region: ORD1
os:
type: windows
resources:
cpu:
# Reference CPU instance label selectors here:
# https://docs.coreweave.com/resources/resource-based-pricing#cpu-only-instance-pricing
type: amd-epyc-rome
count: 4
memory: 16Gi
storage:
root:
size: 80Gi
storageClassName: block-nvme-ord1
source:
pvc:
namespace: vd-images
# Reference querying source image here:
# https://docs.coreweave.com/virtual-servers/root-disk-lifecycle-management/exporting-coreweave-images-to-a-writable-pvc#identifying-source-image
name: winsvr2022std-master-20220319-ord1
# Change user name and pasword
users:
- username:
password:
network:
directAttachLoadBalancerIP: true
public: false
initializeRunning: true
cloudInit: |
autologon: false
parsec: false
edid: false
affinity:
podAntiAffinity:
requiredDuringSchedulingIgnoredDuringExecution:
- labelSelector:
matchExpressions:
- key: app.kubernetes.io/component
operator: In
values:
- dc
topologyKey: topology.kubernetes.io/zone

Then, kubectl is used to deploy it using the create command:

kubectl create -f virtual-server-windows-internal-ip-only.yaml

The kubectl get pods --watch is used to watch the Virtual Server creation progress.

After allowing about five minutes after the Virtual Server has reached the Running status so that the machine may complete its initial start procedures, the External IP is acquired using get vs. Virtual Servers are Custom Resource Definitions, which may be targeted by kubectl using vs.

kubectl get vs

Install and configure domain services

With the Virtual Server up and running, SSH is used to connect using the Internal IP acquired using kubectl get vs. Connections via RDP are also possible, if a graphical interface is preferred.

The following PowerShell script installs and configures the Domain Services role.

Click to expand - cw_adds_setup.ps1
$DomainName = Read-Host -Prompt "Enter desired Domain Name"
$Tenant = Read-Host -Prompt "Enter CoreWeave tenant name"

winrm quickconfig -q

Add-WindowsFeature AD-Domain-Services -IncludeManagementTools

Import-Module ADDSDeployment
Install-ADDSForest `
-CreateDnsDelegation:$false `
-DatabasePath "C:\Windows\NTDS" `
-DomainMode "WinThreshold" `
-DomainName "$($DomainName).$($Tenant).svc.tenant.chi.local" `
-DomainNetbiosName $($DomainName) `
-ForestMode "WinThreshold" `
-InstallDns:$true `
-LogPath "C:\Windows\NTDS" `
-NoRebootOnCompletion:$false `
-SysvolPath "C:\Windows\SYSVOL" `
-Force:$true

This script is simply written to an outfile called cw_adds_setup.ps1.

When executed, the script prompts the user to provide the following:

PromptDescription
Domain NameThe name of the Active Directory Domain
CoreWeave Tenant NameThe name of the CoreWeave tenant - usually tenant-<orgname>-<namespace>. Used to integrate your DNS Suffix with Kubernetes Core DNS
SafeModeAdministratorPasswordUsed for Directory Services Restore Mode

When the script is finished executing, the server will automatically reboot as part of the ADDS deployment.

In this example, the server is constructed with the following attributes:

AttributeValue
Domain NameAD
Search Realmad.tenant-orgname-namespace.svc.tenant.chi.local
PDC/DNS Server IP10.135.123.123
PDC FQDNvs-pdc.ad.tenant-orgname-namespace.svc.tenant.chi.local

The Windows Server is now fully provisioned!

Join another Windows Virtual Server

First, another Windows Virtual Server is provisioned using the same procedure as the first. This second server is configured to point to our Primary Domain Controller (or PDC):

Set-DnsClientServerAddress -InterfaceAlias 'Ethernet' -ServerAddresses 10.135.123.123

The domain is then added using Add-Computer:

Add-Computer -DomainName ad.tenant-orgname-namespace.svc.tenant.chi.local

At this point, the user is prompted for credentials. The user account must have domain join permissions on your domain.

After rebooting the second server, this Windows Virtual Server will now be joined to the Active Directory Domain. Connectivity can be confirmed by performing a Group Policy update:

gpupdate /force

Add a secondary Domain Controller

To create an additional domain controller, the procedure for provisioning a Primary Domain Controller is repeated.

The PowerShell script below is used to install the Domain Services role and configure the new Domain Controller.

Click to expand - cw_addc_setup.ps1
$DomainName = Read-Host -Prompt "Enter Domain Name"
$Tenant = Read-Host -Prompt "Enter CoreWeave tenant name"
Write-Host "Ensure to precede username with $($domainname+'\')" -ForegroundColor Red -BackgroundColor Black
$usr = Read-Host "Domain Admin UserName"
$passwd= Read-Host "Domain Admin Password" -AsSecureString
$cred = new-object System.Management.Automation.PSCredential($usr,$passwd)

winrm quickconfig -q

Add-WindowsFeature AD-Domain-Services -IncludeManagementTools

Import-Module ADDSDeployment
Install-ADDSDomainController `
-NoGlobalCatalog:$false `
-CreateDnsDelegation:$false `
-Credential $cred `
-CriticalReplicationOnly:$false `
-DatabasePath "C:\Windows\NTDS" `
-DomainName "$($DomainName).$($Tenant).svc.tenant.chi.local" `
-InstallDns:$true `
-LogPath "C:\Windows\NTDS" `
-NoRebootOnCompletion:$false `
-SiteName "Default-First-Site-Name" `
-SysvolPath "C:\Windows\SYSVOL" `
-Force:$true

As before, the script is printed to an output file, and then executed afterwards.

When executed, the script prompts the user to provide the following:

PromptDescription
Domain NameThe name of the Active Directory Domain
CoreWeave Tenant NameThe name of the CoreWeave tenant - usually tenant-<orgname>-<namespace>. Used to integrate your DNS Suffix with Kubernetes Core DNS
SafeModeAdministratorPasswordUsed for Directory Services Restore Mode

The server will automatically reboot. After rebooting, the Domain Controller status is confirmed using Get-ADDomainController: