--- title: Using a Service to Expose Your App weight: 10 --- ## {{% heading "objectives" %}} * Learn about a Service in Kubernetes. * Understand how labels and selectors relate to a Service. * Expose an application outside a Kubernetes cluster. ## Overview of Kubernetes Services Kubernetes [Pods](/docs/concepts/workloads/pods/) are mortal. Pods have a [lifecycle](/docs/concepts/workloads/pods/pod-lifecycle/). When a worker node dies, the Pods running on the Node are also lost. A [Replicaset](/docs/concepts/workloads/controllers/replicaset/) might then dynamically drive the cluster back to the desired state via the creation of new Pods to keep your application running. As another example, consider an image-processing backend with 3 replicas. Those replicas are exchangeable; the front-end system should not care about backend replicas or even if a Pod is lost and recreated. That said, each Pod in a Kubernetes cluster has a unique IP address, even Pods on the same Node, so there needs to be a way of automatically reconciling changes among Pods so that your applications continue to function. {{% alert %}} _A Kubernetes Service is an abstraction layer which defines a logical set of Pods and enables external traffic exposure, load balancing and service discovery for those Pods._ {{% /alert %}} A [Service](/docs/concepts/services-networking/service/) in Kubernetes is an abstraction which defines a logical set of Pods and a policy by which to access them. Services enable a loose coupling between dependent Pods. A Service is defined using YAML or JSON, like all Kubernetes object manifests. The set of Pods targeted by a Service is usually determined by a _label selector_ (see below for why you might want a Service without including a `selector` in the spec). Although each Pod has a unique IP address, those IPs are not exposed outside the cluster without a Service. Services allow your applications to receive traffic. Services can be exposed in different ways by specifying a `type` in the `spec` of the Service: * _ClusterIP_ (default) - Exposes the Service on an internal IP in the cluster. This type makes the Service only reachable from within the cluster. * _NodePort_ - Exposes the Service on the same port of each selected Node in the cluster using NAT. Makes a Service accessible from outside the cluster using `NodeIP:NodePort`. Superset of ClusterIP. * _LoadBalancer_ - Creates an external load balancer in the current cloud (if supported) and assigns a fixed, external IP to the Service. Superset of NodePort. * _ExternalName_ - Maps the Service to the contents of the `externalName` field (e.g. `foo.bar.example.com`), by returning a `CNAME` record with its value. No proxying of any kind is set up. This type requires v1.7 or higher of `kube-dns`, or CoreDNS version 0.0.8 or higher. More information about the different types of Services can be found in the [Using Source IP](/docs/tutorials/services/source-ip/) tutorial. Also see [Connecting Applications with Services](/docs/tutorials/services/connect-applications-service/). Additionally, note that there are some use cases with Services that involve not defining a `selector` in the spec. A Service created without `selector` will also not create the corresponding Endpoints object. This allows users to manually map a Service to specific endpoints. Another possibility why there may be no selector is you are strictly using `type: ExternalName`. ## Services and Labels A Service routes traffic across a set of Pods. Services are the abstraction that allows pods to die and replicate in Kubernetes without impacting your application. Discovery and routing among dependent Pods (such as the frontend and backend components in an application) are handled by Kubernetes Services. Services match a set of Pods using [labels and selectors](/docs/concepts/overview/working-with-objects/labels), a grouping primitive that allows logical operation on objects in Kubernetes. Labels are key/value pairs attached to objects and can be used in any number of ways: * Designate objects for development, test, and production * Embed version tags * Classify an object using tags {{< figure src="/docs/tutorials/kubernetes-basics/public/images/module_04_labels.svg" class="diagram-medium" >}} Labels can be attached to objects at creation time or later on. They can be modified at any time. Let's expose our application now using a Service and apply some labels. ### Step 1: Creating a new Service Let’s verify that our application is running. We’ll use the `kubectl get` command and look for existing Pods: ```shell kubectl get pods ``` If no Pods are running then it means the objects from the previous tutorials were cleaned up. In this case, go back and recreate the deployment from the [Using kubectl to create a Deployment](/docs/tutorials/kubernetes-basics/deploy-app/deploy-intro#deploy-an-app) tutorial. Please wait a couple of seconds and list the Pods again. You can continue once you see the one Pod running. Next, let’s list the current Services from our cluster: ```shell kubectl get services ``` We have now a running Service called kubernetes-bootcamp. Here we see that the Service received a unique cluster-IP, an internal port and an external-IP (the IP of the Node). To find out what port was opened externally (for the `type: NodePort` Service) we’ll run the `describe service` subcommand: ```shell kubectl describe services/kubernetes-bootcamp ``` Create an environment variable called `NODE_PORT` that has the value of the Node port assigned: ```shell export NODE_PORT="$(kubectl get services/kubernetes-bootcamp -o go-template='{{(index .spec.ports 0).nodePort}}')" echo "NODE_PORT=$NODE_PORT" ``` Now we can test that the app is exposed outside of the cluster using `curl`, the IP address of the Node and the externally exposed port: ```shell curl http://"$(minikube ip):$NODE_PORT" ``` {{< note >}} If you're running minikube with Docker Desktop as the container driver, a minikube tunnel is needed. This is because containers inside Docker Desktop are isolated from your host computer. In a separate terminal window, execute: ```shell minikube service kubernetes-bootcamp --url ``` The output looks like this: ``` http://127.0.0.1:51082 ! Because you are using a Docker driver on darwin, the terminal needs to be open to run it. ``` Then use the given URL to access the app: ```shell curl 127.0.0.1:51082 ``` {{< /note >}} And we get a response from the server. The Service is exposed. ### Step 2: Using labels The Deployment created automatically a label for our Pod. With the `describe deployment` subcommand you can see the name (the _key_) of that label: ```shell kubectl describe deployment ``` Let’s use this label to query our list of Pods. We’ll use the `kubectl get pods` command with `-l` as a parameter, followed by the label values: ```shell kubectl get pods -l app=kubernetes-bootcamp ``` You can do the same to list the existing Services: ```shell kubectl get services -l app=kubernetes-bootcamp ``` Get the name of the Pod and store it in the POD_NAME environment variable: ```shell export POD_NAME="$(kubectl get pods -o go-template --template '{{range .items}}{{.metadata.name}}{{"\n"}}{{end}}')" echo "Name of the Pod: $POD_NAME" ``` To apply a new label we use the label subcommand followed by the object type, object name and the new label: ```shell kubectl label pods "$POD_NAME" version=v1 ``` This will apply a new label to our Pod (we pinned the application version to the Pod), and we can check it with the `describe pod` command: ```shell kubectl describe pods "$POD_NAME" ``` We see here that the label is attached now to our Pod. And we can query now the list of pods using the new label: ```shell kubectl get pods -l version=v1 ``` And we see the Pod. ### Step 3: Deleting a service To delete Services you can use the `delete service` subcommand. Labels can be used also here: ```shell kubectl delete service -l app=kubernetes-bootcamp ``` Confirm that the Service is gone: ```shell kubectl get services ``` This confirms that our Service was removed. To confirm that route is not exposed anymore you can `curl` the previously exposed IP and port: ```shell curl http://"$(minikube ip):$NODE_PORT" ``` This proves that the application is not reachable anymore from outside of the cluster. You can confirm that the app is still running with a `curl` from inside the pod: ```shell kubectl exec -ti $POD_NAME -- curl http://localhost:8080 ``` We see here that the application is up. This is because the Deployment is managing the application. To shut down the application, you would need to delete the Deployment as well. ## {{% heading "whatsnext" %}} * Tutorial [Running Multiple Instances of Your App](/docs/tutorials/kubernetes-basics/scale/scale-intro/). * Learn more about [Service](/docs/concepts/services-networking/service/).