--- title: "Example: Deploying WordPress and MySQL with Persistent Volumes" assignees: - ahmetb --- {% capture overview %} This tutorial shows you how to deploy a WordPress site and a MySQL database using Minikube. Both applications use PersistentVolumes and PersistentVolumeClaims to store data. A [PersistentVolume](/docs/concepts/storage/persistent-volumes/) (PV) is a piece of storage in the cluster that has been provisioned by an administrator, and a [PeristentVolumeClaim](/docs/concepts/storage/persistent-volumes/#persistentvolumeclaims) (PVC) is a set amount of storage in a PV. PersistentVolumes and PeristentVolumeClaims are independent from Pod lifecycles and preserve data through restarting, rescheduling, and even deleting Pods. **Warning:** This deployment is not suitable for production use cases, as it uses single instance WordPress and MySQL Pods. Consider using [WordPress Helm Chart](https://github.com/kubernetes/charts/tree/master/stable/wordpress) to deploy WordPress in production. {: .warning} {% endcapture %} {% capture objectives %} * Create a PersistentVolume * Create a Secret * Deploy MySQL * Deploy WordPress * Clean up {% endcapture %} {% capture prerequisites %} {% include task-tutorial-prereqs.md %} Download the following configuration files: 1. [local-volumes.yaml](/docs/tutorials/stateful-application/mysql-wordpress-persistent-volume/local-volumes.yaml) 1. [mysql-deployment.yaml](/docs/tutorials/stateful-application/mysql-wordpress-persistent-volume/mysql-deployment.yaml) 1. [wordpress-deployment.yaml](/docs/tutorials/stateful-application/mysql-wordpress-persistent-volume/wordpress-deployment.yaml) {% endcapture %} {% capture lessoncontent %} ## Create a PersistentVolume MySQL and Wordpress each use a PersistentVolume to store data. While Kubernetes supports many different [types of PersistentVolumes](/docs/concepts/storage/persistent-volumes/#types-of-persistent-volumes), this tutorial covers [hostPath](/docs/concepts/storage/volumes/#hostpath). **Note:** If you have a Kubernetes cluster running on Google Container Engine, please follow [this guide](https://cloud.google.com/container-engine/docs/tutorials/persistent-disk). {: .note} ### Setting up a hostPath Volume A `hostPath` mounts a file or directory from the host node’s filesystem into your Pod. **Warning:** Only use `hostPath` for developing and testing. With hostPath, your data lives on the node the Pod is scheduled onto and does not move between nodes. If a Pod dies and gets scheduled to another node in the cluster, the data is lost. {: .warning} 1. Launch a terminal window in the directory you downloaded the manifest files. 2. Create two PersistentVolumes from the `local-volumes.yaml` file: kubectl create -f local-volumes.yaml {% include code.html language="yaml" file="mysql-wordpress-persistent-volume/local-volumes.yaml" ghlink="/docs/tutorials/stateful-application/mysql-wordpress-persistent-volume/local-volumes.yaml" %} {:start="3"} 3. Run the following command to verify that two 20GiB PersistentVolumes are available: kubectl get pv The response should be like this: NAME CAPACITY ACCESSMODES RECLAIMPOLICY STATUS CLAIM STORAGECLASS REASON AGE local-pv-1 20Gi RWO Retain Available 1m local-pv-2 20Gi RWO Retain Available 1m ## Create a Secret for MySQL Password A [Secret](/docs/concepts/configuration/secret/) is an object that stores a piece of sensitive data like a password or key. The manifest files are already configured to use a Secret, but you have to create your own Secret. 1. Create the Secret object from the following command: kubectl create secret generic mysql-pass --from-literal=password=YOUR_PASSWORD **Note:** Replace `YOUR_PASSWORD` with the password you want to apply. {: .note} 2. Verify that the Secret exists by running the following command: kubectl get secrets The response should be like this: NAME TYPE DATA AGE mysql-pass Opaque 1 42s **Note:** To protect the Secret from exposure, neither `get` nor `describe` show its contents. {: .note} ## Deploy MySQL The following manifest describes a single-instance MySQL Deployment. The MySQL container mounts the PersistentVolume at /var/lib/mysql. The `MYSQL_ROOT_PASSWORD` environment variable sets the database password from the Secret. {% include code.html language="yaml" file="mysql-wordpress-persistent-volume/mysql-deployment.yaml" ghlink="/docs/tutorials/stateful-application/mysql-wordpress-persistent-volume/mysql-deployment.yaml" %} 1. Deploy MySQL from the `mysql-deployment.yaml` file: kubectl create -f mysql-deployment.yaml 2. Verify that the Pod is running by running the following command: kubectl get pods **Note:** It can take up to a few minutes for the Pod's Status to be `RUNNING`. {: .note} The response should be like this: NAME READY STATUS RESTARTS AGE wordpress-mysql-1894417608-x5dzt 1/1 Running 0 40s ## Deploy WordPress The following manifest describes a single-instance WordPress Deployment and Service. It uses many of the same features like a PVC for persistent storage and a Secret for the password. But it also uses a different setting: `type: NodePort`. This setting exposes WordPress to traffic from outside of the cluster. {% include code.html language="yaml" file="mysql-wordpress-persistent-volume/wordpress-deployment.yaml" ghlink="/docs/tutorials/stateful-application/mysql-wordpress-persistent-volume/wordpress-deployment.yaml" %} 1. Create a WordPress Service and Deployment from the `wordpress-deployment.yaml` file: kubectl create -f wordpress-deployment.yaml 2. Verify that the Service is running by running the following command: kubectl get services wordpress The response should be like this: NAME CLUSTER-IP EXTERNAL-IP PORT(S) AGE wordpress 10.0.0.89 80:32406/TCP 4m **Note:** Minikube can only expose Services through `NodePort`.

The `EXTERNAL-IP` is always ``. {: .note} 3. Run the following command to get the IP Address for the WordPress Service: minikube service wordpress --url The response should be like this: http://1.2.3.4:32406 4. Copy the IP address, and load the page in your browser to view your site. You should see the WordPress set up page similar to the following screenshot. ![wordpress-init](https://raw.githubusercontent.com/kubernetes/examples/master/mysql-wordpress-pd/WordPress.png) **Warning:** Do not leave your WordPress installation on this page. If another user finds it, they can set up a website on your instance and use it to serve malicious content.

Either install WordPress by creating a username and password or delete your instance. {: .warning} {% endcapture %} {% capture cleanup %} 1. Run the following command to delete your Secret: kubectl delete secret mysql-pass 2. Run the following commands to delete all Deployments and Services: kubectl delete deployment -l app=wordpress kubectl delete service -l app=wordpress 3. Run the following commands to delete the PersistentVolumeClaims and the PersistentVolumes: kubectl delete pvc -l app=wordpress kubectl delete pv local-pv-1 local-pv-2 **Note:** Any other Type of PersistentVolume would allow you to recreate the Deployments and Services at this point without losing data, but `hostPath` loses the data as soon as the Pod stops running. {: .note} {% endcapture %} {% capture whatsnext %} * Learn more about [Introspection and Debugging](/docs/tasks/debug-application-cluster/debug-application-introspection/) * Learn more about [Jobs](/docs/concepts/workloads/controllers/jobs-run-to-completion/) * Learn more about [Port Forwarding](/docs/tasks/access-application-cluster/port-forward-access-application-cluster/) * Learn how to [Get a Shell to a Container](/docs/tasks/debug-application-cluster/get-shell-running-container/) {% endcapture %} {% include templates/tutorial.md %}