OSDFIR Infrastructure provides an easy way to install and manage open source Digital Forensics & Incident Response (DFIR) tools on Kubernetes.
This guide explains how to troubleshoot common issues related to OSDFIR Infrastructure’s deployment using kubectl. A Kubernetes dashboard can optionally be deployed to troubleshoot the same concepts, but will not be covered as part of this guide.
The following are the most common issues that users face when dealing with deployments:
To check if the status of your pods is healthy, the easiest way is to run the kubectl get pods command. After that, you can use kubectl describe and kubectl logs to obtain more detailed information.
Once you run kubectl get pods, you may find your pods showing any of the following statuses:
If you see any of the statuses above, this means that the pod could not be scheduled on a node. Usually, this is because of insufficient CPU or memory resources. It may also come up due to a network related issue or provisioning a volume.
IMPORTANT: Please wait a few minutes for errors to resolve when installing a Helm chart for the first time as it is likely still provisioning the necessary resources.
To confirm the cause, note the pod you want to investigate from kubectl get pods, replacing POD-NAME
with the name of the pod, then run:
$ kubectl describe pod POD-NAME
You should see an output message providing some information about why the pod is failing. Additionally, you can review logs of the pod by running:
$ kubectl logs POD-NAME
Another option can be to review logs stored within the pod. To gain shell access to a pod, run:
$ kubectl exec -it POD-NAME -- /bin/bash
TIP: Turbinia logs can be found in
/mnt/turbiniavolume/logs
and Timesketch logs can be found in/var/log/timesketch
.
When your pod status is ImagePullBackOff or ErrImagePull, this means that the pod could not run because it could not pull the image. This typically occurs when there is either an invalid Image or the Kubernetes cluster does not have Internet connectivity. Try pulling the image manually on the node using docker pull. For example, to manually pull an image from Docker Hub, use the command below by replacing IMAGE with the image ID:
$ docker pull IMAGE
TIP: When using a private GKE cluster in GCP, please ensure you have a Cloud NAT configured with the network router of your cluster so that third party dependencies can be retrieved.
When your pod status is Init:CrashLoopBackOff or Init:Error, this means that the pod’s init container could not run. To investigate, review the logs of the init container, replacing POD-NAME
with the pod name and INIT-CONTAINER
with the name of the init pod, then run:
$ kubectl logs POD-NAME -c INIT-CONTAINER
TIP: The respective init container name for Turbinia is
init-turbinia
and for Timesketch isinit-timesketch
.
For more troubleshooting tips, see the official k8s docs for troubleshooting pods.
Sometimes, when you start a new installation in Kubernetes you may find that a service doesn’t respond when you try to access it, although it was created by a pod in a deployment.
First you can check that the service you are trying to access actually exists. To do so, grab a list of services by running kubectl get services, then run the following command (replace SVC-NAME with the name of the service you want to access):
$ kubectl get SVC-NAME
To get detailed information around the service if it exists, run:
$ kubectl describe SVC-NAME
You should see an output message providing some information including any service issues.
If the service is registered and no issues were detected from the step above, it could also be a DNS problem. DNS problems typically occurs when a service is in a different namespace then the pod or if the cluster does not have DNS enabled.
To troubleshoot, confirm that DNS is enabled for your Kubernetes cluster and follow the official k8s documentation to learn how to debug DNS resolution.
Lastly, if the service was attached to an ingress (Load Balancer), you can check for issues related to the ingress deployment. To troubleshoot, run the kubectl get ingress command to get the name of your ingress. Then, replacing INGRESS-NAME
placeholder with the ingress name, run:
$ kubectl describe ingress INGRESS-NAME
You should see an output message providing some information including any ingress issues.
For more troubleshooting tips, see the official k8s docs for troubleshooting services.
The next common problem to occur is Persistent Volumes failing to provision. Persistence in Kubernetes can be explained by three main components:
In order to troubleshoot, first check the status of your Storage Class by running:
$ kubectl get sc
$ kubectl describe sc
You should see an output message providing some information including any Storage Class issues.
If there are no issues reported by investigating the Storage Class, the next step is to check the Persistent Volume (PV) by running:
$ kubectl get pv
$ kubectl describe pv
You should see an output message providing some information including any Persistent Volume issues and whether it is still pending.
If there are no issues in the Persistence Volume, the final place to check is the Persistent Volume Claim (PVC):
$ kubectl get pvc
$ kubectl describe pvc
Similar to above, you should see output providing some information including any Persistent Volume Claim issues and whether it is still pending. Note the Volume and cross reference it with the output from kubectl get pv.
If the Persistent Volume is still pending and no issues were found from the steps above, please ensure you are using a supported volume. A more involved option would be to manually configure a Persistent Volume, or a Storage Class that supports dynamic provisioning.
The following are the most common issues users face when installing Helm charts:
When a timeout occurs installing a chart, the error will look similar to this:
Error: INSTALLATION FAILED: Kubernetes cluster unreachable: Get "https://IP-ADDRESS/version": dial tcp IP-ADDRESS connect: connection timed out
To fix, wait a few minutes then try again. If the problem persists, ensure you are connected to the cluster by running a command such as kubectl get pods or increase the --timeout
field when installing the chart.
Another common issue is credential related errors, caused by a known issue with PostgreSQL authentication failing after re-deployment. This occurs when you delete the deployed Helm chart while keeping the existing persistent volumes as they are not removed by default. When re-deploying the chart using the same release name, this causes the secret generating the credentials to go out-of-sync with the password being persisted.
To fix, ensure the underlying Persistent Volume is removed when uninstalling the chart or use a different release name. For example, looking for a string containing postgresql
, run kubectl get pvc to note the Persistent Volume Claim and kubectl get pv to note the Persistent Volume of the Postgresql deployment. Then replacing PVC-NAME
and PV-NAME
with the respective names, delete the Persistent Volumes by running:
$ kubectl delete pvc PVC-NAME
$ kubectl delete pv PV-NAME
Another option can be to delete all Persistent Volumes in the cluster by running:
$ kubectl delete pvc --all
$ kubectl delete pv --all
If persisting the existing data is important, you can also choose to rollback the previous deployment. Use the following guide for more instructions on rolling back your deployment.
Some other common troubleshooting methods when working with Helm charts is to use the --dry-run
flag and compare the intended state vs the actual state. To do so, replacing RELEASE-NAME
with a name of your release, run:
$ helm install RELEASE-NAME --dry-run --debug ../mychart
Visit the official docs to learn more about Helm and the various commands that can be run.
The following table lists common kubectl commands used when troubleshooting deployment issues. For an extensive list, please refer to the official kubectl cheat sheet.
# Get Cluster events
kubectl events # List all cluster events
kubectl events --types=Warning # List all warning events
kubectl events | grep "error" # List all events and grep for errors
# Get commands with basic output
kubectl get pods -A -o wide # List all pods in all namespaces and associated nodes
kubectl get pods # List all pods in the namespace
kubectl get deployment # List all deployments in the namespace
kubectl get services # List all services in the namespace
kubectl get ingress # List all ingress in the namespace
kubectl get configmap # List all configmaps in the namespace
kubectl get secrets # List all secrets in the namespace
kubectl get hpa # List horizontal autoscaler
kubectl get pvc # List all persistent volume claims in the namespace
kubectl get pv # List all persistent volumes in the namespace
kubectl get sc # List all storage classes
kubectl get nodes # List all nodes in the cluster
# Describe commands with basic output
kubectl describe pod POD-NAME # Detailed information of a given pod
kubectl describe deployment DEPLOYMENT-NAME # Detailed information of a given deployment
kubectl describe service SVC-NAME # Detailed information of a given service
kubectl describe ingress INGRESS-NAME # Detailed information of a given ingress
kubectl describe configmap CONFIG-NAME # Detailed information of a given configmap
kubectl describe secret SECRET-NAME # Detailed information of a given secret
kubectl describe hpa HPA-NAME # Detailed information of a given horizontal autoscaler
kubectl describe pvc PVC-NAME # Detailed information of a given persistent volume claim
kubectl describe pv PV-NAME # Detailed information of a given persistent volume
kubectl describe sc SC-NAME # Detailed information of a given storage class
kubectl describe node NODE-NAME # Detailed information of a given node
# Interacting with running pods
kubectl logs POD-NAME # Show pod logs (stdout)
kubectl logs POD-NAME --previous # Show pod logs (stdout) for a previous instantiation of a container
kubectl logs POD-NAME -c my-container # Show pod container logs of an init container (stdout)
kubectl exec -it POD-NAME -- /bin/bash # Interactive shell access to a running pod
kubectl port-forward POD-NAME 5000:6000 # Listen on port 5000 on the local machine and forward to port 6000 on POD-NAME
kubectl cp /tmp/foo_dir POD-NAME:/tmp/bar_dir # Copy /tmp/foo_dir local directory to /tmp/bar_dir in a remote pod in the current namespace
kubectl cp POD-NAME:/tmp/foo /tmp/bar # Copy /tmp/foo from a remote pod to /tmp/bar locally
kubectl top pods # Show pod usage (cpu/mem)
kubectl top nodes # Show node usage (cpu/mem)
If you are running into an issue that is not covered in this troubleshooting guide, please file a bug and include as much information as possible and we’ll take a look.