Sharing Our Passion for Technology
& Continuous Learning
<   Back to Blog

A Hands-On Tour of Kubernetes: Part 2 - Namespaces and Labels

kubernetes logo on a blue background

Namespaces

We’ve only created one pod so far. Kubernetes wouldn’t be very special if we could only run one pod, so let’s try running multiple pods.

$ kubectl run app-1 --image=nginx:1.24
pod/app-1 created

$ kubectl run app-2 --image=nginx:1.24
pod/app-2 created

$ kubectl run app-3 --image=nginx:1.24
pod/app-3 created

It seems like Kubernetes was happy to create three pods. Let’s list our pods to verify.

$ kubectl get pods
NAME    READY   STATUS    RESTARTS   AGE
app-1   1/1     Running   0          83s
app-2   1/1     Running   0          66s
app-3   1/1     Running   0          17s

Consider that a given Kubernetes cluster may be used to run dozens of applications. What would this list look like if I created hundreds of pods?

Well, there’s no fancy truncation. This list would be very long. Fortunately, Kubernetes allows us to create namespaces so that we can group related resources together.

A namespace is like any other resource in Kubernetes, meaning we can use kubectl to list our namespaces.

$ kubectl get namespaces
NAME              STATUS   AGE
default           Active   8d
kube-system       Active   8d
kube-public       Active   8d
kube-node-lease   Active   8d

It looks like our local cluster already has more than one namespace!

  • The namespaces prefixed with kube- contain resources used by the cluster itself.
  • The default namespace is where new resources will go unless otherwise specified.

Since we haven’t been specifying a namespace in our kubectl commands, all the pods we’ve been creating have been added under the default namespace. Let’s go ahead and create a new namespace for ourselves.

$ kubectl create namespace app
namespace/app created

We can list our namespaces again to verify our new namespace exists.

$ kubectl get namespaces             
NAME              STATUS   AGE
default           Active   8d
kube-system       Active   8d
kube-public       Active   8d
kube-node-lease   Active   8d
app               Active   42s

There it is! Let’s now create a pod under this namespace.

$ kubectl run app-4 --image=nginx:1.24 --namespace=app
pod/app-4 created

Note the --namespace option at the end of the above command. --namespace can be added to most kubectl commands to specify which namespace to use. For example, here is how we can list our newly-created pod:

$ kubectl get pods --namespace=app
NAME    READY   STATUS    RESTARTS   AGE
app-4   1/1     Running   0          7s

Without --namespace we’d be listing the pods in the default namespace. In fact, removing --namespace is the same as explicitly setting the namespace to default.

$ kubectl get pods --namespace=default
NAME    READY   STATUS    RESTARTS   AGE
app-1   1/1     Running   0          19m
app-2   1/1     Running   0          19m
app-3   1/1     Running   0          18m

$ kubectl get pods
NAME    READY   STATUS    RESTARTS   AGE
app-1   1/1     Running   0          19m
app-2   1/1     Running   0          19m
app-3   1/1     Running   0          18m

Many kubectl flags and options provide a long form (e.g. --namespace) and a short form (-n). The short form requires fewer keystrokes, but I will use the long form in this series since it is more descriptive.

When introducing new options, I will show the short form (if one exists) alongside the long form, so feel free to use the version you prefer.


Resource names need to be unique in a given namespace, but different namespaces can have resources with identical names.

For example, since a pod called app-1 exists in the default namespace, we cannot create another pod called app-1 in the default namespace:

$ kubectl run app-1 --image=nginx:1.24
Error from server (AlreadyExists): pods "app-1" already exists

However, we can certainly create a pod called app-1 in the app namespace:

$ kubectl run app-1 --image=nginx:1.24 --namespace=app
pod/app-1 created

We’ll see this name duplication if we list list out the pods in both namespaces:

$ kubectl get pods --namespace=app
NAME    READY   STATUS    RESTARTS   AGE
app-4   1/1     Running   0          5m5s
app-1   1/1     Running   0          15s

$ kubectl get pods --namespace=default
NAME    READY   STATUS    RESTARTS   AGE
app-1   1/1     Running   0          24m
app-2   1/1     Running   0          24m
app-3   1/1     Running   0          23m

It’s important to emphasize that the app-1 pod in the default namespace and the app-1 pod in the app namespace are two different pods. They share nothing other than the name.

Beyond providing a convenient mechanism for organizing resources, namespaces are also central to Kubernetes’ RBAC model and for controlling resource allocation. We won’t go into detail about either of these topics, but these characteristics of namespaces are what facilitate the usage of multi-tenant clusters (multiple teams using the same cluster). However, even for single-tenant clusters, it’s common to create a namespace for every workload.

You can also set the default namespace with

$ kubectl config set-context --current --namespace <namespace>

Deleting a namespace will automatically delete all resources in that namespace. Let’s delete our app namespace and see this in action.

$ kubectl delete namespace app
namespace "app" deleted

Recall that this namespace contained two pods. If we try to list the pods in this namespace, we’ll see that no such resources exist.

$ kubectl get pods --namespace=app
No resources found in app namespace.

Indeed, our namespace is gone entirely.

$ kubectl get namespaces
NAME              STATUS   AGE
default           Active   8d
kube-system       Active   8d
kube-public       Active   8d
kube-node-lease   Active   8d

Before moving on, let’s delete the other pods we created in the default namespace.

$ kubectl delete pod/app-1 pod/app-2 pod/app-3
pod "app-1" deleted
pod "app-2" deleted
pod "app-3" deleted

Labels

Namespaces are an important mechanism for grouping related resources, but they often aren’t sufficient for keeping things organized. What if we want to further segment resources in a namespace? What if we want to group together resources across namespaces?

In Kubernetes, labels allow us to tag resources with arbitrary key-value pairs. We can then query resources by their labels.

For example, let’s say we have three applications. These three applications are creatively named:

  1. app-1
  2. app-2
  3. app-3

All three of these applications consist of:

  • a backend process that exposes a JSON API
  • a frontend process that serves HTML/CSS/JS and consumes the API

Futhermore, suppose these applications have the following visibility:

  • “app-1” and “app-2” are public, internet-facing applications
  • “app-3” is an internal-facing application for support

Let’s set up this fictional environment in our cluster and see how we might label these resources. To start, we’ll create some namespaces for ourselves.

$ kubectl create namespace app-1
namespace/app-1 created

$ kubectl create namespace app-2
namespace/app-2 created

$ kubectl create namespace app-3
namespace/app-3 created

Next, we’ll create the pods representing the backend and frontend processes for each application.

$ kubectl run app-1-backend --image=nginx:1.24 --namespace=app-1
pod/app-1-backend created

$ kubectl run app-1-frontend --image=nginx:1.24 --namespace=app-1
pod/app-1-frontend created

$ kubectl run app-2-backend --image=nginx:1.24 --namespace=app-2
pod/app-2-backend created

$ kubectl run app-2-frontend --image=nginx:1.24 --namespace=app-2
pod/app-2-frontend created

$ kubectl run app-3-backend --image=nginx:1.24 --namespace=app-3
pod/app-3-backend created

$ kubectl run app-3-frontend --image=nginx:1.24 --namespace=app-3
pod/app-3-frontend created

For good measure, we’ll list the pods in each namespace and verify everything is in the right spot.

$ kubectl get pods --namespace=app-1
NAME             READY   STATUS    RESTARTS   AGE
app-1-backend    1/1     Running   0          89s
app-1-frontend   1/1     Running   0          80s

$ kubectl get pods --namespace=app-2
NAME             READY   STATUS    RESTARTS   AGE
app-2-backend    1/1     Running   0          68s
app-2-frontend   1/1     Running   0          61s

$ kubectl get pods --namespace=app-3
NAME             READY   STATUS    RESTARTS   AGE
app-3-backend    1/1     Running   0          56s
app-3-frontend   1/1     Running   0          51s

Without any other changes, how can I list all the public-facing pods? Well, I can’t. I would already need to know the visibility of each application and run multiple kubectl commands.

However, we can remedy this situation by adding a visibility label to our pods. We can then query for pods by their visibility value.

Let’s start by adding the proper visibility label to one of our pods.

$ kubectl label pod/app-1-frontend visibility=public --namespace=app-1
pod/app-1-frontend labeled

kubectl reported our pod was labeled, but does anything look different if we list the pods?

$ kubectl get pods --namespace=app-1
NAME             READY   STATUS    RESTARTS   AGE
app-1-backend    1/1     Running   0          6m34s
app-1-frontend   1/1     Running   0          6m25s

The output looks the same as before, other than updated ages. By default, kubectl get doesn’t show resource labels. We can ask kubectl to show the value of a certain label by using the --label-columns (-L) option:

$ kubectl get pods --label-columns=visibility --namespace=app-1
NAME             READY   STATUS    RESTARTS   AGE   VISIBILITY
app-1-backend    1/1     Running   0          12m   
app-1-frontend   1/1     Running   0          12m   public

We now see a “visibility” column, along with the value of that label for both pods.

  • app-1-frontend shows the value “public”. This is what we set it to previously.
  • app-1-backend shows no value. We haven’t set this label on this pod.

If we don’t know what labels exist on our resources, we can use the --show-labels option to show all the labels that exist on a resource. The output formatting can get a little messy with this command if our resources have many labels, but it’s useful for exploration.

$ kubectl get pods --show-labels --namespace=app-1
NAME             READY   STATUS    RESTARTS   AGE   LABELS
app-1-backend    1/1     Running   0          14m   run=app-1-backend
app-1-frontend   1/1     Running   0          13m   run=app-1-frontend,visibility=public

So it seems our pods already have a run label! This label is added automatically when we use the kubectl run command to create pods. We won’t use this run label for anything, but be aware that certain actions will add labels to our resources.

Let’s go ahead and finish adding the correct visibility label to our other pods.

$ kubectl label pod/app-1-backend visibility=public --namespace=app-1
pod/app-1-backend labeled

$ kubectl label pod/app-2-backend pod/app-2-frontend visibility=public --namespace=app-2
pod/app-2-backend labeled
pod/app-2-frontend labeled

$ kubectl label pod/app-3-backend pod/app-3-frontend visibility=internal --namespace=app-3
pod/app-3-backend labeled
pod/app-3-frontend labeled

Alright, back to our original concern… how do we list all the pods that are public facing? Since our pods are now sufficiently labeled, we can use the --selector (-l) option to provide a label selector to our kubectl get command.

$ kubectl get pods --selector=visibility=public --namespace=app-1
NAME             READY   STATUS    RESTARTS   AGE
app-1-frontend   1/1     Running   0          21m
app-1-backend    1/1     Running   0          21m

Hm, that was disappointing. We only received two pods, but we expect to see four. The --namespace option is still narrowing the query to our app-1 namespace. If we want to query across all namespaces, do we remove it?

$ kubectl get pods --selector=visibility=public
No resources found in default namespace.

Nope! Remember, not specifying --namespace is the same as --namespace=default, so the previous command tried to list all pods matching the giving label selector in the default namespace.

For what we’re trying to accomplish, we need to use --all-namespaces (-A) option.

$ kubectl get pods --selector=visibility=public --all-namespaces
NAMESPACE   NAME             READY   STATUS    RESTARTS   AGE
app-1       app-1-frontend   1/1     Running   0          23m
app-1       app-1-backend    1/1     Running   0          23m
app-2       app-2-backend    1/1     Running   0          22m
app-2       app-2-frontend   1/1     Running   0          22m

Much better! None of the “app-3” pods are included in the output. We can list those “app-3” pods by altering the value of the visibility label in our label selector.

$ kubectl get pods --selector=visibility=internal --all-namespaces
NAMESPACE   NAME             READY   STATUS    RESTARTS   AGE
app-3       app-3-backend    1/1     Running   0          24m
app-3       app-3-frontend   1/1     Running   0          24m

This example shows how we can use labels for simple, ad-hoc analysis, but labels can also be consumed by automated processes to address things like resource auditing and cost analysis. Depending on your organization’s needs, tools like Kyverno and OPA Gatekeeper can enforce the usage of certain labels.

Notably, as we’ll see in upcoming material, labels are also used by other Kubernetes resources to configure their behavior.

If you need to delete a label for any reason, you can use the following command.

kubectl label <type>/<name> <label_name>-

For example, to delete the visibility label from the app-1-frontend pod, we would run:

$ kubectl label pod/app-1-frontend visibility- --namespace=app-1
pod/app-1-frontend unlabeled

Before moving on, let’s delete the resources we’ve created. Remember that deleting a namespace automatically deletes all the resources under that namespace.

$ kubectl delete namespace/app-1 namespace/app-2 namespace/app-3
namespace "app-1" deleted
namespace "app-2" deleted
namespace "app-3" deleted

Imperative vs. Declarative

So far, we’ve been using kubectl to imperatively create our resources. This has been useful for getting our feet off the ground, but Kubernetes generally favors a declarative approach. Rather than telling Kubernetes how we want to create our resources, we tell Kubernetes what to create.

Kubernetes resources are typically declared as YAML manifests, although JSON also works. Let’s look at an example pod manifest.

apiVersion: v1
kind: Pod
metadata:
  name: example
  namespace: default
spec:
  containers:
  - name: example
    image: nginx:1.24

Alright, so there is a bit to unpack here. Let’s break down the meaning of these fields.

  • Together, apiVersion and kind specify the type of resource declared in this manifest. Every resource manifest includes these fields.
    • In our example, we’re using apiVersion: v1, which represents the core API resources. Resources are versioned inside an API group.
    • The kind refers to a specific resource type inside the API version. Since we want a pod, kind is set to Pod. Resource kinds use PascalCase.
  • The metadata object contains various fields that are common across resources.
    • name: The name of the resource. Every resource needs a name.
    • namespace: Which namespace this resource belongs in. If this is not specified, the default namespace is used.
    • Although not shown in the example, metadata also contains fields to hold our labels, annotations, finalizers, and owner references, among others.
  • As the name implies, the spec object contains the resource specification. The resource type determines which fields are included under spec.
    • Resources typically have a set of required fields alongside a set of optional fields.
    • For a pod, we need to specify at least one container. Our example defines a single container with the name example that uses the image nginx:1.24. Note that the container name doesn’t necessarily need to match the pod name, although pods created using kubectl run set the pod and container name to the same value.
    • Many other fields can be set in a pod spec, but they are not included in the example because they are either not required or they assume a default value if unspecified.

Right now, this manifest is just words on a website. Somehow, we need to send this manifest to our cluster. Here is how we do that:

  1. Copy the contents of the manifest into a new file.
  2. Send the manifest to Kubernetes using kubectl apply -f <filename>.

Here is how that might look with bash:

$ cat <<EOF >pod.yaml
apiVersion: v1
kind: Pod
metadata:
  name: example
  namespace: default
spec:
  containers:
  - name: example
    image: nginx:1.24
EOF

$ kubectl apply -f pod.yaml 
pod/example created

And the same, but with PowerShell:

$ @" 
apiVersion: v1
kind: Pod
metadata:
  name: example
  namespace: default
spec:
  containers:
  - name: example
    image: nginx:1.24
"@ > pod.yaml

$ kubectl apply -f pod.yaml
pod/example created

For comparison, here is how we’d create the same pod using imperative commands:

$ kubectl run example --image=nginx:1.24 --namespace=default

On the surface, the imperative approach seems much simpler. Why go through the effort of creating a file and composing a resource manifest when we’re going to run a kubectl command anyway?

Despite the verbosity, the declarative approach has a couple big advantages over imperative resource creation:

  • Repeatability: When deploying your resources with a series of imperative commands (e.g. in a script), there is an implicit assumption that the environment doesn’t change between runs. For certain environments this may be a safe assumption, but for “living” environments such as many production systems, this constraint is rarely satisfied. The declarative approach embraces a live environment – rather than trying to push our desires into the environment, we’ll change the environment to satisfy our desires.
  • Maintainability: Applications change. New components are added to the system. Implementing these changes imperatively involves an ever-evolving set of scripts and/or continuous deployment pipelines. In contrast, the declarative approach has a simple deployment model: point kubectl to the directory containing your manifests and it will ship the resources to the cluster. Should those manifests change, Kubernetes will respond appropriately.

The Kubernetes documentation provides some guidance on when to use which technique (imperative vs. declarative), but for brevity, we’ll continue using imperative commands for the remainder of these posts.


Tip: We can append --dry-run=client -o yaml to our imperative commands to view the manifest of the underlying resource being created.

$ kubectl run example --image=nginx:1.24 --namespace=default --dry-run=client -o yaml
apiVersion: v1
kind: Pod
metadata:
  creationTimestamp: null
  labels:
    run: example
  name: example
  namespace: default
spec:
  containers:
  - image: nginx:1.24
    name: example
    resources: {}
  dnsPolicy: ClusterFirst
  restartPolicy: Always
status: {}

And if you prefer JSON, you can use -o json instead of -o yaml.

Deleting declarative resources is a straightforward process with kubectl using the command:

kubectl delete -f filename.yml

When you run this command, kubectl will delete the resources specified in filename.yml. However, keep in mind that any resources that are not specified in the file will not be deleted.


In this post, we learned about using namespaces, labels, and manifest files to organize and manage our resources. In the next post, we’ll look at how pods communicate with each other and the outside world.