Thursday, January 20, 2022

Kubernetes Troubleshooting

Kubernetes Troubleshooting

View Console logs of a Container (stdout)

kubectl logs pod flaskpod-6db57ff68b-ckklj



Tail Console Logs of a Container

kubectl logs --follow flaskpod-557d6745c7-qkd5v




View recent logs (events) of a container

Kubectl describe po/nginx-7f4558cf8-6jjg2

Kubectl describe pod pod-name

Shows at the bottom the recent events


Or


Kubectl logs myPod -previous



Execute a command within a container

kubectl exec -it pod/mysql-7d8f78bf86-clcg5 /bin/bash


Tip: sometimes not all bash commands are available within a container, but can sometimes be very useful - e.g. if it is a MySql container you can run mysql -ppassword to run SQL commands (where password = password).



Launch busybox in a Namespace

Yaml for busybox: 

apiVersion: v1
kind: Pod
metadata:
name: busybox
namespace: default
spec:
containers:
- image: busybox
command:
- sleep
- "3600"
imagePullPolicy: IfNotPresent
name: busybox
restartPolicy: Always

kubectl apply -f jumpod.yml

pod/busybox created


kubectl exec -it busybox /bin/sh



View detailed deployment info

Kubectl get deploy -o wide



View the rollout history of a deployment

Kubectl rollout history deployment blah 



Get pods in all Namespaces

Kubectl get pod -A

Is get in all namespaces



Apply a deployment file

Kubectl apply -f deployment.yaml 


Create a namespace

kubectl create namespace monitoring





Terraform 101

 Terraform 101

Terraform is used for Infrastructure as Code (IaC).

Terraform is Declarative, NOT Imperative, and defines the DESIRED STATE.

What does that mean? For initial infrastructure creation, they will be very similar (e.g. both will say create 5 servers), but when you update your infrastructure there are differences.

For example, say you have 10 servers, and you want to remove 3...

Imperative commands to update your infrastructure would look like:

- remove 3 servers

Declarative commands to update your infrastructure would look like:

- have 7 servers


Terraform will work out what it needs to do to achieve the Desired state.

With Terraform, you re-run the whole configuration file to achieve your update.

Terraform Stages

refresh - query infra to get current state

plan - create an execution plan (plan to achieve the desired state, based on the current state). No changes.

apply - execute the plan

destroy - removes whole setup, removing elements in the correct order. E.g. remove servers before remove VPC. Destroy also runs a Refresh, and then a Plan before removing the resources.

Terraform plugin for VSCode

https://marketplace.visualstudio.com/items?itemName=HashiCorp.terraform


Install Terraform on Mac

First install Homebrew - https://brew.sh/

https://learn.hashicorp.com/tutorials/terraform/install-cli

brew tap hashicorp/tap

brew install hashicorp/tap/terraform

brew update

brew upgrade hashicorp/tap/terraform


Terraform Basic Operation

In vscode, create a new directory - e.g. Documents\terraform

create the file: main.tf

provider "aws" {
region = "us-east-1"
access_key = "access-key"
secret_key = "your-secret-key"
}

resource "aws_vpc" "myvpc" {
cidr_block = "10.0.0.0/16"
}


This will create a VPC resource - for syntax see:

https://registry.terraform.io/providers/hashicorp/aws/latest/docs/resources/vpc


In VSCode, click the 'View' menu and select "Terminal".

Now, run the command:

terraform init



Initializing the backend... Initializing provider plugins... - Reusing previous version of hashicorp/aws from the dependency lock file - Using previously-installed hashicorp/aws v3.72.0 Terraform has been successfully initialized! You may now begin working with Terraform. Try running "terraform plan" to see any changes that are required for your infrastructure. All Terraform commands should now work. If you ever set or change modules or backend configuration for Terraform, rerun this command to reinitialize your working directory. If you forget, other commands will detect it and remind you to do so if necessary.

Terraform has successfully initialised the plugin for AWS.

terraform plan



The terraform plan command examines the current state, and works out what needs to be changed in order to achieve the desired state.

In the main.tf file we specified a VPC with CIDR 10.0.0.0/16, and here terraform can see that it currently doesn't exist - hence the (known after apply) lines.
For example the arn (amazon resource name) for the VPC is unknown because we haven't created it yet.


Terraform Apply


Again, the apply command examines the current state, and says what it will do in order to achieve the desired state.
Type 'yes' to continue to perform the actions.




Terraform Destroy


This will destroy the resources listed in your main.tf file.


Terraform will examine the current state, and will say what actions it will take to destroy the resources.
Type 'yes' to continue.




State File

The state file is called 'terraform.tfstate'. This contains the current state.
There is also a back file called 'terraform.tfstate.backup', but this isn't the latest.

When you run an plan, apply, or destroy command, the terraform.tfstate file will be updated. The old contents of the file will be copied to the backup file.

Variables

Defined in the main.tf file.

Variables are a way of setting a value which can be used multiple times.
Variables can be a string, number, boolean, list, map, tuple, or object.

String Variables











Number Variables










Boolean Variables








List Variables

Lists can contain strings, numbers, or boolean values.







Note for lists, like an array, the first element is accessed by item number 0.
The value of item 0 = 'Value1'

Map Variables

Maps are key-value pairs.

The elements of a map are accessed using the key you gave,
For example:














When you access key1, its value is 'Value1'.

Tuples

Tuples are similar to Lists, but can contain different data types.

variable "mytuple" {
type = tuple([string, number, string])
default = ["cat",1,"dog"]
}


Objects

variable "myobject"{
type = object({name = string, port = list(number)})
default = {
name = "fw"
port = [22,25,80]
}
}


Using Variables

Variables are referenced using 'var.'
variable "vpcname" {
type = string
default = "myvpc"
}

resource "aws_vpc" "myvpc" {
cidr_block = "10.0.0.0/16"
tags = {
Name = var.vpcname
}
}


Or to access a list variable:
variable "mylist" {
type = list(string)
default = [ "Value1","Value2" ]
}

resource "aws_vpc" "myvpc" {
cidr_block = "10.0.0.0/16"
tags = {
Name = var.mylist[0]
}
}


Or to access a map variable:
variable "mymap" {
type = map
default = {
key1 = "Value1"
key2 = "Value2"
}
}

resource "aws_vpc" "myvpc" {
cidr_block = "10.0.0.0/16"
tags = {
Name = var.mymap["key1"]
}
}

Note, the key name is in quotes!


Input Variables

Allows user to manually specify a value when running terraform plan
variable "inputname" {
type = string
description = "set the name of the vpc"
}

resource "aws_vpc" "myvpc" {
cidr_block = "10.0.0.0/16"
tags = {
Name = var.inputname
}
}

Now when running terraform plan, you will see:







Outputs

After Terraform creates a resource, you may want to know some information about it such as the arn or id. Note, this will only work with terraform apply.

To see which attributes can be fetched, see the Terraform documentation, e.g gor AWS VPC resource, the attributes are:  https://registry.terraform.io/providers/hashicorp/aws/latest/docs/resources/vpc

resource "aws_vpc" "myvpc" {
cidr_block = "10.0.0.0/16"
tags = {
Name = var.inputname
}
}
output "vpcid" {
value = aws_vpc.myvpc.id
}
output "vpcarn" {
value = aws_vpc.myvpc.arn
}







Note: the outputs get saved to the State file terraform.tfstate


Creating an EC2 instance

The documentation for EC2 instance is 'aws_instance'


main.tf
provider "aws" {
region = "us-east-1"
}

resource "aws_instance" "ec2instance" {
ami = "ami-08e4e35cccc6189f4"
instance_type = "t2.micro"
}

output "ec2_arn" {
value = aws_instance.ec2instance.arn
}














Docker 101

 Docker 101

Install Docker for Mac

https://docs.docker.com/get-docker/

Create a Container

To create a container in docker, you need to first create a folder.
Within this folder you will need 3 docker files, and the files (code) to run your app.
The docker files are:
  • docker-compose.yaml
  • requirements.txt
  • Dockerfile

The following example here is a flask container, which uses a python script (app.py).
We have 4 files, all located in the same directory. The name of the directory doesn't matter.

app.py
Dockerfile
requirements.txt
docker-compose.yaml

Dockerfile

FROM python:3.8

WORKDIR /app
COPY . .

RUN pip install -r requirements.txt

ENTRYPOINT ["python"]
CMD ["app.py"]

The first thing here is the image which is used as the base image - denoted by the line "FROM python:3.8". Phyton 3.8 is a standard image containing Python. We then add to this image to install additional Python libraries before the program is executed - in this case 'pip install -r requirements.txt'.
Finally the 'Dockerfile' contains what should be executed when the container loads (python app.py). 

requirements.txt

Flask==0.12
flask-restful==0.3.5
flask-mysqldb==0.2.0
mysqlclient==1.4.2.post1

The requirements.txt file simply includes the python libraries which should be installed before the program is executed. This is the same as you would run on your PC - e.g pip install Flask==0.12.

docker-compose.yaml

version: "3.7"

services:
helloworld:
build:
context: ./
ports:
- 5000:5000

The docker-compose.yaml file states how the network requirements for the application - how it should be exposed to the outside world.


We now run the command:
docker compose build
from the directory which includes the 4 files mentioned above.

This will build the container, first downloading the image for python (python:3.8), and then installing the additional python libraries mentioned in the requirements.txt file. The docker image will include all of the files in the directory from where you ran the 'docker compose build' command.

The docker image will be automatically imported to docker running locally on your PC. To view it, run:
docker images

The name of the image = the name of the folder from where you ran the 'docker compose build' command concatenated with and underscore and the name of the service from the docker-compose.yaml file. In this case, my folder name is 'test' and the service name is 'helloworld'. The image name is therefore 'test_helloworld'.

Next, you want to upload the image to docker hub. To do that you first need to setup an account (free) on dockerhub https://hub.docker.com.

In order to upload the image to docker hub you must first login with a browser and create a repository on https://hub.docker.com. Make a note of the name of your repository. You can create a public or private repository. For this example I used a public repository. 

Next, you need to tag your image with your dockerhub username.

If your username is 'my-username', the repository name is 'my-repo', and the image created by 'docker compose build' is named 'testhelloworld', the tag command would be:

docker tag test_helloworld my-username/myrepo:aUniqueTagName

Note 'aUniqueTagName' is used to identify the image you are uploading and will be used in pull requests (downloads) against docker hub. The ':aUniqueTagName' is optional for the docker tag command, but I strongly recommend using it!

If you run docker images again, you will see your tagged image there.

Before uploading the image to docker hub, you must login to docker hub on the command line. To do that, run:


docker login -u your-username -p your-password docker.io


You can now upload your image to docker hub using:


docker push my-username/myrepo:aUniqueTagName



If you are using the image for a kubernetes deployment, in your deployment.yaml file, the image would be referenced as:
spec:
containers:
- name: name-of-your-container
image: docker.io/my-username/myrepo:aUniqueTagName




Other Docker commands

show running containers

docker ps 

show running and stopped containers

docker ps -a