Skip to content

Latest commit

 

History

History
374 lines (307 loc) · 11.3 KB

03-compute-resources.md

File metadata and controls

374 lines (307 loc) · 11.3 KB

Provisioning Compute Resources

Kubernetes requires a set of machines to host the Kubernetes control plane and the worker nodes where containers are ultimately run. In this lab you will provision the compute resources required for running a secure and highly available Kubernetes cluster.

Ensure a default region has been set as described in the Prerequisites lab.

Resource Group

In Azure, compute resources are tied to a resource group, let's create one for the tutorial:

az group create --name k8s-the-hard-way --location eastus

output

{
  "id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/k8s-the-hard-way",
  "location": "eastus",
  "managedBy": null,
  "name": "k8s-the-hard-way",
  "properties": {
    "provisioningState": "Succeeded"
  },
  "tags": null,
  "type": "Microsoft.Resources/resourceGroups"
}

Networking

The Kubernetes networking model assumes a flat network in which containers and nodes can communicate with each other. In cases where this is not desired network policies can limit how groups of containers are allowed to communicate with each other and external network endpoints.

Setting up network policies is out of scope for this tutorial.

Virtual Network

In this section a dedicated Virtual Network (VNET) will be setup to host the Kubernetes cluster. A subnet must be provisioned with an IP address range large enough to assign a private IP address to each node in the Kubernetes cluster.

Create the kubernetes-the-hard-way custom VNET network and subnet:

az network vnet create \
  --resource-group k8s-the-hard-way \
  --name k8s-the-hard-way-vnet \
  --address-prefixes 10.240.0.0/16 \
  --subnet-name k8s-the-hard-way-subnet \
  --subnet-prefixes 10.240.0.0/24

The 10.240.0.0/24 IP address range can host up to 254 compute instances.

Security Group

Create a security group that allows internal communication across all protocols:

az network nsg create --resource-group k8s-the-hard-way --name k8s-the-hard-way-nsg

Create a firewall rule that allows external SSH, and HTTPS:

az network nsg rule create \
  --resource-group k8s-the-hard-way \
  --nsg-name k8s-the-hard-way-nsg \
  --name K8s \
  --access Allow \
  --protocol Tcp \
  --direction Inbound \
  --priority 100 \
  --source-address-prefix "*" \
  --source-port-range "*" \
  --destination-port-ranges 22 6443

An external load balancer will be used to expose the Kubernetes API Servers to remote clients.

List the security group rule in the k8s-the-hard-way resource group:

az network nsg rule show --resource-group k8s-the-hard-way --name K8s --nsg-name k8s-the-hard-way-nsg

output

{
  "access": "Allow",
  "description": null,
  "destinationAddressPrefix": "*",
  "destinationAddressPrefixes": [],
  "destinationApplicationSecurityGroups": null,
  "destinationPortRange": null,
  "destinationPortRanges": [
    "22",
    "6443"
  ],
  "direction": "Inbound",
  "etag": "W/\"716618b2-6703-49f8-a488-844de09522a6\"",
  "id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/k8s-the-hard-way/providers/Microsoft.Network/networkSecurityGroups/k8s-the-hard-way-nsg/securityRules/K8s",
  "name": "K8s",
  "priority": 100,
  "protocol": "Tcp",
  "provisioningState": "Succeeded",
  "resourceGroup": "k8s-the-hard-way",
  "sourceAddressPrefix": "*",
  "sourceAddressPrefixes": [],
  "sourceApplicationSecurityGroups": null,
  "sourcePortRange": "*",
  "sourcePortRanges": [],
  "type": "Microsoft.Network/networkSecurityGroups/securityRules"
}

Kubernetes Public IP Address

Allocate a static IP address that will be attached to the external load balancer fronting the Kubernetes API Servers:

az network public-ip create \
  --name k8s-the-hard-way-ip \
  --resource-group k8s-the-hard-way \
  --allocation-method Static

Verify the k8s-the-hard-way static IP address was created in your default compute region:

az network public-ip show --resource-group k8s-the-hard-way --name k8s-the-hard-way-ip

output

{
  "ddosSettings": null,
  "deleteOption": null,
  "dnsSettings": null,
  "etag": "W/\"c1cf04cc-9c30-4f57-b90d-a44062699064\"",
  "extendedLocation": null,
  "id": "/subscriptions/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx/resourceGroups/k8s-the-hard-way/providers/Microsoft.Network/publicIPAddresses/k8s-the-hard-way-ip",
  "idleTimeoutInMinutes": 4,
  "ipAddress": "20.120.83.207",
  "ipConfiguration": null,
  "ipTags": [],
  "linkedPublicIpAddress": null,
  "location": "eastus",
  "migrationPhase": null,
  "name": "k8s-the-hard-way-ip",
  "natGateway": null,
  "provisioningState": "Succeeded",
  "publicIpAddressVersion": "IPv4",
  "publicIpAllocationMethod": "Static",
  "publicIpPrefix": null,
  "resourceGroup": "k8s-the-hard-way",
  "resourceGuid": "08d52962-3f4c-4112-8f25-515474bc9c71",
  "servicePublicIpAddress": null,
  "sku": {
    "name": "Basic",
    "tier": "Regional"
  },
  "tags": null,
  "type": "Microsoft.Network/publicIPAddresses",
  "zones": null
}

The Kubernetes Frontend Load Balancer

In this section you will provision an external load balancer to front the Kubernetes API Servers. The k8s-the-hard-way-ip static IP address will be attached to the resulting load balancer.

The compute instances created in this tutorial will not have permission to complete this section. Run the following commands from the same machine used to create the compute instances.

Provision a Network Load Balancer

Create the external load balancer network resources:

az network lb create \
  --name k8s-the-hard-way-lb \
  --resource-group k8s-the-hard-way \
  --backend-pool-name k8s-the-hard-way-lb-pool \
  --public-ip-address k8s-the-hard-way-ip
az network lb probe create \
  --lb-name k8s-the-hard-way-lb \
  --resource-group k8s-the-hard-way \
  --name k8s-the-hard-way-lb-probe \
  --port 80 \
  --protocol tcp
az network lb rule create \
  --resource-group k8s-the-hard-way \
  --lb-name k8s-the-hard-way-lb \
  --name k8s-the-hard-way-lb-rule \
  --protocol tcp \
  --frontend-port 6443 \
  --backend-port 6443 \
  --backend-pool-name k8s-the-hard-way-lb-pool \
  --probe-name k8s-the-hard-way-lb-probe  

Compute Instances

The compute instances in this lab will be provisioned using Ubuntu Server 20.04, which has good support for the containerd container runtime. Each compute instance will be provisioned with a fixed private IP address to simplify the Kubernetes bootstrapping process.

Kubernetes Controllers

Create three network interfaces and three compute instances (in an availability set) which will host the Kubernetes control plane:

for i in 0 1 2; do
  az network public-ip create \
    --name controller-${i}-ip \
    --resource-group k8s-the-hard-way \
    --allocation-method Static
done
for i in 0 1 2; do
  az network nic create \
    --resource-group k8s-the-hard-way \
    --name controller-${i}-nic \
    --vnet-name k8s-the-hard-way-vnet \
    --subnet k8s-the-hard-way-subnet \
    --network-security-group k8s-the-hard-way-nsg \
    --public-ip-address controller-${i}-ip \
    --private-ip-address 10.240.0.1${i} \
    --lb-name k8s-the-hard-way-lb \
    --lb-address-pools k8s-the-hard-way-lb-pool \
    --ip-forwarding true
done
az vm availability-set create --name k8s-the-hard-way-as -g k8s-the-hard-way 
for i in 0 1 2; do
  az vm create \
    --name controller-${i} \
    --resource-group k8s-the-hard-way \
    --availability-set k8s-the-hard-way-as \
    --no-wait \
    --nics controller-${i}-nic \
    --image UbuntuLTS \
    --admin-username azureuser \
    --generate-ssh-keys \
    --size Standard_B2s \
    --data-disk-sizes-gb 200
done

Kubernetes Workers

Each worker instance requires a pod subnet allocation from the Kubernetes cluster CIDR range. The pod subnet allocation will be used to configure container networking in a later exercise. The pod-cidr instance metadata will be used to expose pod subnet allocations to compute instances at runtime.

The Kubernetes cluster CIDR range is defined by the Controller Manager's --cluster-cidr flag. In this tutorial the cluster CIDR range will be set to 10.200.0.0/16, which supports 254 subnets.

Create three compute instances which will host the Kubernetes worker nodes:

for i in 0 1 2; do
  az network public-ip create \
    --name worker-${i}-ip \
    --resource-group k8s-the-hard-way \
    --allocation-method Static
done
for i in 0 1 2; do
  az network nic create \
    --resource-group k8s-the-hard-way \
    --name worker-${i}-nic \
    --vnet-name k8s-the-hard-way-vnet \
    --subnet k8s-the-hard-way-subnet \
    --network-security-group k8s-the-hard-way-nsg \
    --public-ip worker-${i}-ip \
    --private-ip-address 10.240.0.2${i} \
    --ip-forwarding true \
    --tags podCidr=10.200.${i}.0/24
done
for i in 0 1 2; do
  az vm create \
    --name worker-${i} \
    --resource-group k8s-the-hard-way \
    --no-wait \
    --nics worker-${i}-nic \
    --image UbuntuLTS \
    --admin-username azureuser \
    --generate-ssh-keys \
    --size Standard_B2s \
    --data-disk-sizes-gb 200 \
    --tags podCidr=10.200.${i}.0/24
done

Verification

List the compute instances in your default compute zone:

az vm list --resource-group k8s-the-hard-way --output table

output

Name          ResourceGroup     Location    Zones
------------  ----------------  ----------  -------
controller-0  k8s-the-hard-way  eastus
controller-1  k8s-the-hard-way  eastus
controller-2  k8s-the-hard-way  eastus
worker-0      k8s-the-hard-way  eastus
worker-1      k8s-the-hard-way  eastus
worker-2      k8s-the-hard-way  eastus

Configuring SSH Access

SSH will be used to configure the controller and worker instances. When building the compute instances, if you don't currently have an SSH keypair, one will be generated for you and stored in your ~/.ssh directory

Let's build an SSH config file to easily be able to SSH to all our controller and worker nodes throughout the lab:

for instance in controller-0 controller-1 controller-2 worker-0 worker-1 worker-2; do
  EXTERNAL_IP=$(az vm show --show-details -g k8s-the-hard-way -n ${instance} --query publicIps --output tsv)
  cat <<EOF | tee -a ~/.ssh/config
  Host ${instance}
  User azureuser
  HostName ${EXTERNAL_IP}
  IdentityFile ~/.ssh/id_rsa
  ServerAliveInterval 120
EOF
done

After this is done, set the correct permissions on the new config file:

chmod 644 ~/.ssh/config

Test SSH access to the controller-0 compute instances:

ssh controller-0
Welcome to Ubuntu 18.04 LTS (GNU/Linux 4.15.0-1006-gcp x86_64)

...

Last login: Sun May 13 14:34:27 2018 from XX.XXX.XXX.XX

Type exit at the prompt to exit the controller-0 compute instance:

$azureuser@controller-0:~$ exit

output

logout
Connection to XX.XXX.XXX.XXX closed

Next: Provisioning a CA and Generating TLS Certificates