Skip to content

Commit

Permalink
Add quick start guide (#1)
Browse files Browse the repository at this point in the history
- Added Quick Start Guide which uses port-forwarding to expose NuoDB Control
  Plane and NuoDB database.
- Added a database injection example for an application provisioned in the same
  cluster
  • Loading branch information
sivanov-nuodb authored Sep 11, 2023
1 parent a6bc26d commit 6b8eca9
Show file tree
Hide file tree
Showing 7 changed files with 462 additions and 2 deletions.
26 changes: 24 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,21 @@

This repository is for releases and documentation for the NuoDB Control Plane, which enables automatic management of NuoDB databases in Kubernetes.

NuoDB Control Plane allows users to provision NuoDB databases on-demand remotely using REST services by exposing various predefined configuration options.
NuoDB domain and database are modeled as Kubernetes [Custom Resources][1] (CRs).
Their [Custom Resource Definitions][2] (CRDs) act as contracts between NuoDB Operator and the other modules.

![nuodb-control-plane-overview](docs/images/overview.png)

## Documentation

The NuoDB Control Plane documentation is available in [docs](./docs) directory of this repository.
The REST API non-interactive documentation is available at [API documentation][3].

### Get Started

- [Quick Start Guide](./docs/QuickStart.md)

## Installation

To install the NuoDB Control Plane into your Kubernetes cluster, first add the Helm repository for the NuoDB Control Plane Helm charts.
Expand Down Expand Up @@ -79,15 +94,22 @@ To enable multi-tenancy, NuoDB _databases_ managed by the NuoDB Control Plane ar
Create a NuoDB database as follows:

1. Create a project using the `PUT projects` REST endpoint:

```sh
curl -X PUT -H 'Content-Type: application/json' \
$BASE_URL/projects/acme/messaging -d '{"sla": "dev", "tier": "n0.small"}'
```

2. Create a database using the `PUT databases` REST endpoint:

```sh
curl -X PUT -H 'Content-Type: application/json' \
$BASE_URL/databases/acme/messaging/demo -d '{"dbaPassword": "dba"}'
```

> *NOTE*: Projects and databases are identified by resource path segments `<organization>/<project>` and `<organization>/<project>/<database>`, respectively.
The example above creates the database `demo` within the project `messaging` within the organization `acme`.
>**Note**
> Projects and databases are identified by resource path segments `<organization>/<project>` and `<organization>/<project>/<database>`, respectively. The example above creates the database `demo` within the project `messaging` within the organization `acme`.
[1]: https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/#custom-resources
[2]: https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/#customresourcedefinitions
[3]: https://nuodb.github.io/nuodb-cp-releases/api-doc
109 changes: 109 additions & 0 deletions docs/DatabaseInjector.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
# Database Injector Guide

NuoDB Control Plane (CP) can be used to create NuoDB databases for applications that are running in the same [Kubernetes][1] cluster.
This document describes how to use database injector to supply database connection information to sample YCSB application.

## Prerequisites

- A running [Kubernetes cluster][2]
- [kubectl][3] installed and able to access the cluster.

## Installing NuoDB Control Plane

Install NuoDB CP as documented in [Installation](../README.md#installation) section.

### Enable Cluster-scoped Access

By default NuoDB operator will monitor only the local namespace for NuoDB [custom resources][4].
If cluster-scoped access is required, set the `cpOperator.watchNamespace.enabled=false` Helm value during installation. E.g.:

```sh
helm install nuodb-cp-operator nuodb-cp/nuodb-cp-operator \
--namespace nuodb-cp-system \
--set cpOperator.watchNamespace.enabled=false \
--set nuodb.serviceAccount.create=false \
--set nuodb.serviceAccount.name=default \
...
```

>**Note**
> The `nuodb` service account (SA) creation is disabled in the above command for simplicity. To enable NuoDB Kubernetes Aware Admin (KAA) capabilities, the NuoDB processes should be given special permissions to access Kubernetes API server.
For more information, please check [Automatic Management of NuoDB State](https://doc.nuodb.com/nuodb/latest/deployment-models/kubernetes-environments/kubernetes-aware-admin/). For cluster-scoped deployments, the NuoDB SA and RBAC should be provisioned before hand in each namespace where NuoDB databases are created.

## Database Injection

NuoDB operator can inject database information into ConfigMap's _data_ once the database is ready.
This enables easy data source configuration in the application container and acts as a dependency mechanism without the need of additional _init_ containers.

Database connection details are populated _after_ the database is created and ready to accept SQL connections which blocks application container creation.
All containers that have references to the target ConfigMap will fail with `CreateContainerConfigError` due to the ConfigMap key being absent.

### Injected Properties

| Property | Description |
| ----- | ----------- |
| `dbName` | NuoDB database name |
| `dbHost` | The FQDN of the domain managing this database. If external access is enabled on the domain, the property will be populated with the external FQDN. |
| `dbPort` | The database port for SQL clients. If external access is enabled on the domain, the property will be populated with the Ingress Controller's service port (by default _443_). |
| `caPem` | The Certificate Authority (CA) certificate for the domain. Used by SQL clients that need to enable TLS encryption on the database connections. If TLS is not enabled on the domain, the property is not injected. |

### Creating Database

NuoDB domain and database resources can be created either via REST API or declaratively using [custom resources][4].
This example creates NuoDB domain and database using custom resources.

```sh
kubectl apply -f https://raw.githubusercontent.com/nuodb/nuodb-cp-releases/main/docs/files/domain.yaml
kubectl apply -f https://raw.githubusercontent.com/nuodb/nuodb-cp-releases/main/docs/files/database.yaml
```

## Creating Sample Application

Create a sample Yahoo! Cloud Serving Benchmark (YCSB) application and reference the database information into the _app_ container.

```sh
kubectl apply -f https://raw.githubusercontent.com/nuodb/nuodb-cp-releases/main/docs/files/ycsb-demo-app.yaml
```

Wait for the database to become ready.

```sh
kubectl wait --for=condition=ready database acme-messaging-demo
```

Verify that database information is injected into `acme-messaging-demo-info` ConfigMap's data and the YCSB Pod is _Running_.

```sh
kubectl get cm acme-messaging-demo-info -o yaml
kubectl get pods -l app=ycsb-load
```

### Injector Configuration

Database injection is controlled by custom annotations specified on the target ConfigMap.

| Annotation | ConfigMap data key | Description | Example Value |
| ----- | ----- | ----------- | ------ |
| `cp.nuodb.com/inject-database` | `cp.inject-database` | Specifies that the object should be injected with database information. The annotation value is a reference to a database object in form of "name" or "namespace/name". If the namespace part is omitted, it is inferred from the target ConfigMap. | `acme-messaging-demo` |
| `cp.nuodb.com/inject-database-properties` | `cp.inject-database-properties` | Specifies the database properties that should be injected into the target object. The value is a comma-delimited string of properties. By default, all connection information properties are injected. | `dbHost,dbName` |
| `cp.nuodb.com/inject-database-<property>-field` | `cp.inject-database-<property>-field` | Specifies the database property to target field mapping. By default the field name will match the property name, e.g. "dbHost" property will be injected as "dbHost" field. See [Injected Properties](#injected-properties). | `"cp.nuodb.com/inject-database-dbHost-field": "host"` |
| `cp.nuodb.com/inject-database-notready` | `cp.inject-database-notready` | Whether a non-ready database should be injected. Only the annotation existence is checked. Its value is currently not used. By default, the database is injected only after it becomes ready. | `true` |

There is an alternative way to configure database injection using the ConfigMap _data_ markers.
Their values are treated the same way as the annotation value.
This can be used in environments where custom annotations can't be specified.

## Cleanup

Delete all resources.

```sh
kubectl delete -f https://raw.githubusercontent.com/nuodb/nuodb-cp-releases/main/docs/files/ycsb-demo-app.yaml
kubectl delete -f https://raw.githubusercontent.com/nuodb/nuodb-cp-releases/main/docs/files/database.yaml
kubectl delete -f https://raw.githubusercontent.com/nuodb/nuodb-cp-releases/main/docs/files/domain.yaml
```

[1]: https://kubernetes.io/docs/home/
[2]: https://kubernetes.io/docs/concepts/overview/components/
[3]: https://kubernetes.io/docs/tasks/tools/
[4]: https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/#custom-resources
198 changes: 198 additions & 0 deletions docs/QuickStart.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,198 @@
# DBaaS Quick Start Guide

This document describes how to provision NuoDB databases in multi-tenancy model by using NuoDB Control Plane (CP).
NuoDB Control Plane works with [Kubernetes][1] locally or in the cloud.
The steps in this guide can be followed regardless of the selected Kubernetes platform provider.

## Prerequisites

- A running [Kubernetes cluster][2]
- [kubectl][3] installed and able to access the cluster.
- [Helm 3.x][4] installed.

## Installing Dependencies

### Install Cert Manager

To enable [admission webhooks][7] in the NuoDB operator, [cert-manager](https://github.com/cert-manager/cert-manager) must be installed to automatically generate certificates for the webhook server.

Add the official Helm repositories.

```sh
helm repo add jetstack https://charts.jetstack.io
helm repo update
```

Install Cert Manager Helm chart.

```sh
helm upgrade --install cert-manager jetstack/cert-manager \
--namespace cert-manager \
--set installCRDs=true \
--create-namespace
```

Wait for Cert Manager to become available.

```sh
kubectl -n cert-manager wait pod --all --for=condition=Ready
```

## Installing NuoDB Control Plane

The NuoDB Control Plane consists of [Custom Resource Definitions][5] and the following workloads:

- *NuoDB CP Operator*, which enforces the desired state of the NuoDB [custom resources][6].
- *NuoDB CP REST service*, that exposes a REST API allowing users to manipulate and inspect DBaaS entities.

By default the NuoDB CP will operate in a single namespace only which will be used for NuoDB CP and all databases created by it.
The databases are grouped into *projects*, which are themselves grouped into *organizations*.

Add the official Helm repositories.

```sh
helm repo add nuodb-cp https://nuodb.github.io/nuodb-cp-releases/charts
helm repo update
```

Install NuoDB CP Helm charts.

```sh
helm upgrade --install nuodb-cp-crd nuodb-cp/nuodb-cp-crd \
--namespace nuodb-cp-system \
--create-namespace

helm upgrade --install nuodb-cp-operator nuodb-cp/nuodb-cp-operator \
--namespace nuodb-cp-system \
--set cpOperator.webhooks.enabled=true \
--set 'cpOperator.extraArgs[0]=--ingress-https-port=48006' # Enables connecting to databases with port-forwarding

helm upgrade --install nuodb-cp-rest nuodb-cp/nuodb-cp-rest \
--namespace nuodb-cp-system \
--set cpRest.authentication.enabled=true \
--set cpRest.authentication.admin.create=true \
--set cpRest.baseDomainName=dbaas.localtest.me # Enables connecting to databases with port-forwarding
```

Wait for NuoDB Control Plane to become available.

```sh
kubectl -n nuodb-cp-system -l app=nuodb-cp-operator wait pod --all --for=condition=Ready
kubectl -n nuodb-cp-system -l app=nuodb-cp-rest wait pod --all --for=condition=Ready
```

## Creating NuoDB Database

Once the Control Plane is deployed, projects and databases can now be created.

### Access and Authentication

This guide will use port forwarding and [cURL][9] to demonstrate how to create projects and databases through the REST service.

```sh
kubectl port-forward -n nuodb-cp-system svc/nuodb-cp-rest 8080 2>&1 >/dev/null &
```

To successfully authenticate with the REST API, get the *system/admin* user's password from the cluster:

```sh
PASS=$(kubectl get secret dbaas-user-system-admin -n nuodb-cp-system -o jsonpath='{.data.password}' | base64 -d)
BASE_URL="http://localhost:8080"
```

### Create Project

Create a new project *messaging* in organization *acme*:

```sh
curl -u "system/admin:$PASS" -X PUT -H 'Content-Type: application/json' \
$BASE_URL/projects/acme/messaging \
-d '{"sla": "dev", "tier": "n0.small"}'
```

>**Note**
> Creating project and database with `n0.small` service tier will require 3 vCPU and 5Gi RAM allocatable resources from your cluster. If your setup is resource constrained, consider using `n0.nano` service tier.
Wait for the project to become available.

```sh
while [ "$(curl -s -u "system/admin:$PASS" $BASE_URL/projects/acme/messaging | jq '.status.ready')" != "true" ]; do echo "Waiting ..."; sleep 5; done; echo "Domain is available"
```

### Create database

Create a new database *demo* in project *messaging*:

```sh
curl -u "system/admin:$PASS" -X PUT -H 'Content-Type: application/json' \
$BASE_URL/databases/acme/messaging/demo \
-d '{"dbaPassword": "secret"}'
```

Wait for the database to become available.

```sh
while [ "$(curl -s -u "system/admin:$PASS" $BASE_URL/databases/acme/messaging/demo | jq '.status.ready')" != "true" ]; do echo "Waiting ..."; sleep 5; done; echo "Database is available"
```

### Connect to Database

This guide will use port forwarding to connect to the NuoDB database.

```sh
ADMIN_SVC=$(kubectl get svc -n nuodb-cp-system \
-l 'cp.nuodb.com/organization=acme,cp.nuodb.com/project=messaging,!cp.nuodb.com/database' -oname | grep "clusterip")
DB_SVC=$(kubectl get svc -n nuodb-cp-system \
-l "cp.nuodb.com/organization=acme,cp.nuodb.com/project=messaging,cp.nuodb.com/database" -oname)
kubectl port-forward -n nuodb-cp-system $ADMIN_SVC 48004 2>&1 >/dev/null &
kubectl port-forward -n nuodb-cp-system $DB_SVC 48006 2>&1 >/dev/null &
```

Connect to the NuoDB database via `nuosql` (requires [nuodb-client][8] package v20230228 or later).

```sh
CA_CERT="$(curl -s -u "system/admin:$PASS" $BASE_URL/databases/acme/messaging/demo | jq -r '.status.caPem')"
DB_URL="$(curl -s -u "system/admin:$PASS" $BASE_URL/databases/acme/messaging/demo | jq -r '.status.sqlEndpoint')"
nuosql "demo@${DB_URL}" --user dba --password secret --connection-property trustedCertificates="$CA_CERT"
```

### Cleanup

- Delete all custom resources that have been created in `nuodb-cp-system` namespace.

```sh
kubectl config set-context --current --namespace=nuodb-cp-system
kubectl get databases.cp.nuodb.com -o name | xargs kubectl delete
kubectl get domains.cp.nuodb.com -o name | xargs kubectl delete
kubectl get servicetiers.cp.nuodb.com -o name | xargs kubectl delete
kubectl get helmfeatures.cp.nuodb.com -o name | xargs kubectl delete
kubectl get databasequotas.cp.nuodb.com -o name | xargs -r kubectl delete
kubectl get secrets -o name --selector=cp.nuodb.com/organization | xargs -r kubectl delete
kubectl get pvc -o name --selector=group=nuodb | xargs -r kubectl delete
```

- Cleanup the installed resources in the following order:

```sh
helm uninstall nuodb-cp-rest --namespace nuodb-cp-system
helm uninstall nuodb-cp-operator --namespace nuodb-cp-system
helm uninstall nuodb-cp-crd --namespace nuodb-cp-system
helm uninstall cert-manager --namespace cert-manager
```

- Delete the provisioned namespace:

```sh
kubectl delete namespace nuodb-cp-system
kubectl delete namespace cert-manager
```

[1]: https://kubernetes.io/docs/home/
[2]: https://kubernetes.io/docs/concepts/overview/components/
[3]: https://kubernetes.io/docs/tasks/tools/
[4]: https://helm.sh/
[5]: https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/#customresourcedefinitions
[6]: https://kubernetes.io/docs/concepts/extend-kubernetes/api-extension/custom-resources/#custom-resources
[7]: https://kubernetes.io/docs/reference/access-authn-authz/extensible-admission-controllers/
[8]: https://github.com/nuodb/nuodb-client/releases
[9]: https://curl.se/
36 changes: 36 additions & 0 deletions docs/files/database.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
---
apiVersion: v1
kind: Secret
metadata:
name: acme-messaging-demo-credentials
type: Opaque
stringData:
dbUser: dba
dbPassword: secret

---
apiVersion: cp.nuodb.com/v1beta1
kind: Database
metadata:
name: acme-messaging-demo
labels:
cp.nuodb.com/organization: "acme"
cp.nuodb.com/project: "messaging"
spec:
version: "5.0"
type:
tierRef:
name: n0.small
namespace: nuodb-cp-system
sla: dev
dbName: demo
domainRef:
name: acme-messaging
passwordRef:
kind: Secret
name: acme-messaging-demo-credentials
dataKey: dbPassword
archiveVolume:
volumeSize: 10Gi
journalVolume:
volumeSize: 5Gi
Loading

0 comments on commit 6b8eca9

Please sign in to comment.