Docker Volume Plugin that enables the management of volumes on Ceph RBD backends. This is a hard fork of https://github.com/yp-engineering/rbd-docker-plugin
This plugin will perform the following:
- The name used on the volume will be used for locating the Ceph image (ex.: mypool/myvolume)
- When creating/removing a volume, it will try to locate an image with that name and perform operations on Ceph cluster
- When mounting a volume to a container, it will try to locate that image, create it if doesn't exist yet, map it to the host, format it using a specified filesystem (xfs is default), mount the device to an directory and Docker will bind that directory to the container
- Only one mapping is permitted per image, so we will perform an exclusive lock on Ceph images to avoid corruption.
Modern Linux Kernel comes with a Ceph module for mapping images as virtual devices on OS. This module is very efficient but it doesn't support recent features on Ceph Images, like journaling and fast-diff. By default this plugin will use the 'rbd-nbd' instead of the kernel module. It does the same mapping as the kernel module, but with a little less performance but supports all Ceph Image features. If you want to force Kernel Module usage, set USE_RBD_KERNEL_MODULE to true.
- Set HOST_IP on .env to your machine IP
- Not for Mac Users: you cannot access the ports exposed when using "network_mode: host", so the best is you to run Docker inside a Virtual Box Linux machine in this case
- Run a sample Ceph Storage Cluster
docker-compose.yml
etcd0:
image: quay.io/coreos/etcd:v3.2.25
network_mode: host
environment:
- ETCD_LISTEN_CLIENT_URLS=http://0.0.0.0:12379
- ETCD_ADVERTISE_CLIENT_URLS=http://${HOST_IP}:12379
mon1:
image: flaviostutz/ceph-monitor:13.2.0.2
network_mode: host
pid: host
environment:
- LOG_LEVEL=0
- CREATE_CLUSTER=true
- ETCD_URL=http://${HOST_IP}:12379
- MONITOR_ADVERTISE_ADDRESS=${HOST_IP}:16789
- MONITOR_BIND_PORT=16789
mgr1:
image: flaviostutz/ceph-manager:13.2.0.2
network_mode: host
pid: host
ports:
- 18443:8443 #dashboard https
- 18003:8003 #restful https
- 19283:9283 #prometheus
environment:
- MONITOR_HOSTS=${HOST_IP}:16789
- ETCD_URL=http://${HOST_IP}:12379
osd1:
image: flaviostutz/ceph-osd:13.2.0.2
network_mode: host
pid: host
environment:
- LOG_LEVEL=0
- MONITOR_HOSTS=${HOST_IP}:16789
- OSD_EXT4_SUPPORT=true
- OSD_JOURNAL_SIZE=512
- OSD_CRUSH_LOCATION=root=default host=host1
- ETCD_URL=http://${HOST_IP}:12379
# - OSD_PUBLIC_IP=${HOST_IP}
# - OSD_CLUSTER_IP=${HOST_IP}
osd2:
image: flaviostutz/ceph-osd:13.2.0.2
network_mode: host
pid: host
environment:
- LOG_LEVEL=0
- MONITOR_HOSTS=${HOST_IP}:16789
- OSD_EXT4_SUPPORT=true
- OSD_JOURNAL_SIZE=512
- OSD_CRUSH_LOCATION=root=default host=host2
- ETCD_URL=http://${HOST_IP}:12379
osd3:
image: flaviostutz/ceph-osd:13.2.0.2
network_mode: host
pid: host
environment:
- LOG_LEVEL=0
- MONITOR_HOSTS=${HOST_IP}:16789
- OSD_EXT4_SUPPORT=true
- OSD_JOURNAL_SIZE=512
- OSD_CRUSH_LOCATION=root=default host=host3
- ETCD_URL=http://${HOST_IP}:12379
- Run Cepher plugin
docker plugin install flaviostutz/cepher \
--grant-all-permissions \
--alias=cepher \
MONITOR_HOSTS="${HOST_IP}:16789,${HOST_IP}:26789,${HOST_IP}:16789" \
ETCD_URL="http://${HOST_IP}:12379" \
DEFAULT_IMAGE_SIZE=100 \
ENABLE_AUTO_CREATE_VOLUMES=true
- Test it!
docker run -it --rm --volume-driver=cepher --name first --volume volumes/myimage:/mnt/foo ubuntu /bin/bash -c "echo -n 'Hello ' >> /mnt/foo/hello"
docker run -it --rm --volume-driver=cepher --name second --volume volumes/myimage:/mnt/foo ubuntu /bin/bash -c "cat /mnt/foo/hello"
- There is a default "volumes" pool that this image prepares for you, but if you wish to create a customized pool, follow these steps
# connect to a container that can be used as a Ceph Client
docker-compose exec mgr1 bash
# create a pool for volume images (each volume in Docker will be an image in RDB)
ceph osd pool create mypool 30
rbd pool init mypool
# set max bytes for this pool (optional)
ceph osd pool set-quota mypool data max_bytes 1000000000
# initialize this pool with rbd
rbd pool init mypool
See more at http://docs.ceph.com/docs/master/start/quick-rbd/ and http://docs.ceph.com/docs/jewel/rados/operations/pools/
- Run your container with a persisted volume in Ceph
# run a container that will make the first usage of the image, so it will be
# created on Ceph during volume creation
# after echoing to a file in the newly created volume, this container will exit
# and remove its instance's data, but not the image data.
docker run -it --rm --volume-driver=cepher --name first --volume volumes/myimage:/mnt/foo ubuntu /bin/bash -c "echo -n 'Hello ' >> /mnt/foo/hello"
# run a second docker container, but use the same volume/image from the previous
# step so that we can see the persisted data
docker run -it --rm --volume-driver=cepher --name second --volume volumes/myimage:/mnt/foo ubuntu /bin/bash -c "cat /mnt/foo/hello"
# on Ceph client machine, list the created image
docker-compose exec mgr1 bash
rbd ls default
Name | Required | Description | Default value |
---|---|---|---|
MONITOR_HOSTS | yes | A comma separated list of monitor-ip:port |
|
ETCD_URL | no | if defined, the plugin will search for a base64 encoded keyring at /[cluster-name]/keyring |
|
CEPH_KEYRING_BASE64 | no | base64 encoded keyring to be used to connect to Ceph Cluster, if not defined, the plugin will search for a base64 encoded keyring at /[cluster-name]/keyring on ETC_URL |
|
CEPH_AUTH | no | none or cephx |
cephx |
CEPH_USER | no | user name to use to connect to Ceph | admin |
CEPH_CLUSTER_NAME | no | Ceph cluster name | ceph |
ENABLE_AUTO_CREATE_VOLUMES | no | whatever this plugin will create new images on Ceph cluster if the corresponding image is not found | false |
ENABLE_AUTO_CREATE_POOLS | no | create new pool on Ceph cluster if the corresponding pool is not found | false |
DEFAULT_IMAGE_SIZE | no | default image size for newly created images. maybe overridden by opt | 100 |
DEFAULT_IMAGE_FS | no | default image filesystem for newly created images. maybe overridden by opt | xfs |
DEFAULT_IMAGE_FEATURES | no | default image features for newly created images. maybe overridden by opt | layering,striping,exclusive-lock,object-map,fast-diff,journaling |
VOLUME_REMOVE_ACTION | no | ignore : does nothing on Ceph Cluster when a volume is deleted; delete : deletes the corresponding image from Ceph Cluster (irreversible!); rename - renames the corresponding Ceph Image to trash_[incremental counter]_[imagename] |
rename |
DEFAULT_POOL_NAME | no | default pool name when not specified in volume name | volumes |
DEFAULT_POOL_CREATE | no | whatever during plugin initialization, it will look for the default pool and create it or not | true |
DEFAULT_POOL_PG_NUM | no | number of PGs for the default pool when creating it | 100 |
DEFAULT_POOL_QUOTA_MAX_BYTES | no | max bytes size for the default pool during creation | |
USE_RBD_KERNEL_MODULE | no | if true, will use the Linux RBD Kernel Module that has greater performance, but doesn't support recent image features. if false, will use official Ceph rbd-nbd tool for mapping the images that supports all recent image features. |
false |
LOG_LEVEL | no | debug, info, warning or error | info |
- pool - name of Ceph pool
- name - name of Ceph image
- size - image size when creating a new image in MB
- fstype - filesystem type to create on newly created images. mkfs.[fstype] must be present in OS
- features - Ceph image features applied to newly created images. defaults to 'layering,striping,exclusive-lock,object-map,fast-diff,journaling'
Sample production deployment setup use this configuration:
- Ubuntu 18.04 with kernel 4.15.0-50 (default)
- Docker 18.09.6
- docker-compose 1.23.2
- Ceph 13.2.5
-
We will use 8 machines on this sample
- Machine 1 (MON): one MON container + one MGR container
- Machine 2 (MON): one MON container
- Machine 3 (MON): one MON container
- Machine 4 (OSD): one OSD container + 50GB disk attached
- Machine 5 (OSD): one OSD container + 50GB disk attached
- Machine 6 (OSD): one OSD container + 50GB disk attached
- Machine 7 (OSD): one OSD container + 50GB disk attached
- Machine 8 (DOCKER): docker daemon with Cepher plugin
-
Preparation
- Create a second network connecting OSD hosts
- On one machine, git clone http://github.com/flaviostutz/cepher
- Edit .env and set IPs accordingly
- Copy files to target machines
-
On Machine 1 (MON):
- run
docker-compose up -f docker-compose-mon1.yml
- run
-
On Machine 2 (MON):
- run
docker-compose up -f docker-compose-mon2.yml
- run
-
On Machine 3 (MON):
- run
docker-compose up -f docker-compose-mon3.yml
- run
-
For each Machines 4-7 (OSD):
- edit docker-compose-osds.yml
- set OSD_IP to the IP of this host in the same network as the other Monitors
- set OSD_CLUSTER_IP to the IP of this host in the secondary network between OSDs
- Prepare storage
- Attach a disk to the host
- Format it to XFS
- Mount disk to '/mnt/osd1-sda' (place in fstab for automatic mounting during boot)
- run
docker-compose up -f docker-compose-osds.yml
- You can execute those exact steps for adding new storage even after the cluster is in use
- edit docker-compose-osds.yml
-
On Machine 8:
docker plugin install flaviostutz/cepher \
--grant-all-permissions \
--alias=cepher \
MONITOR_HOSTS="${MON1_IP}:16789,${MON2_IP}:26789,${MON3_IP}:36789" \
ETCD_URL="http://${ETCD_IP}:12379" \
DEFAULT_IMAGE_SIZE=100 \
ENABLE_AUTO_CREATE_VOLUMES=true
-
Validation
- On Machine 1 (MON1)
docker-compose exec mgr1 bash
ceph -s
#check for status
- On Machine 8 (DOCKER)
docker run -it --rm --volume-driver=cepher --name first --volume volumes/myimage:/mnt/foo ubuntu /bin/bash -c "echo -n 'Hello ' >> /mnt/foo/hello"
docker run -it --rm --volume-driver=cepher --name second --volume volumes/myimage:/mnt/foo ubuntu /bin/bash -c "cat /mnt/foo/hello"