forked from vitalif/vitastor
Compare commits
3 Commits
master
...
rdma-zeroc
Author | SHA1 | Date |
---|---|---|
Vitaliy Filippov | eb9fc274e8 | |
Vitaliy Filippov | 9681b62204 | |
Vitaliy Filippov | 8faf8f7b58 |
|
@ -2,6 +2,6 @@ cmake_minimum_required(VERSION 2.8)
|
||||||
|
|
||||||
project(vitastor)
|
project(vitastor)
|
||||||
|
|
||||||
set(VERSION "0.6.10")
|
set(VERSION "0.6.2")
|
||||||
|
|
||||||
add_subdirectory(src)
|
add_subdirectory(src)
|
||||||
|
|
123
README-ru.md
123
README-ru.md
|
@ -22,6 +22,7 @@ Vitastor на данный момент находится в статусе п
|
||||||
|
|
||||||
Однако следующее уже реализовано:
|
Однако следующее уже реализовано:
|
||||||
|
|
||||||
|
0.5.x (стабильная версия):
|
||||||
- Базовая часть - надёжное кластерное блочное хранилище без единой точки отказа
|
- Базовая часть - надёжное кластерное блочное хранилище без единой точки отказа
|
||||||
- Производительность ;-D
|
- Производительность ;-D
|
||||||
- Несколько схем отказоустойчивости: репликация, XOR n+1 (1 диск чётности), коды коррекции ошибок
|
- Несколько схем отказоустойчивости: репликация, XOR n+1 (1 диск чётности), коды коррекции ошибок
|
||||||
|
@ -40,31 +41,26 @@ Vitastor на данный момент находится в статусе п
|
||||||
- Драйвер диска для QEMU (собирается вне дерева исходников QEMU)
|
- Драйвер диска для QEMU (собирается вне дерева исходников QEMU)
|
||||||
- Драйвер диска для утилиты тестирования производительности fio (также собирается вне дерева исходников fio)
|
- Драйвер диска для утилиты тестирования производительности fio (также собирается вне дерева исходников fio)
|
||||||
- NBD-прокси для монтирования образов ядром ("блочное устройство в режиме пользователя")
|
- NBD-прокси для монтирования образов ядром ("блочное устройство в режиме пользователя")
|
||||||
- Утилита для удаления образов/инодов (vitastor-cli rm-data)
|
- Утилита удаления образов/инодов (vitastor-rm)
|
||||||
- Пакеты для Debian и CentOS
|
- Пакеты для Debian и CentOS
|
||||||
|
|
||||||
|
0.6.x (master-ветка):
|
||||||
- Статистика операций ввода/вывода и занятого места в разрезе инодов
|
- Статистика операций ввода/вывода и занятого места в разрезе инодов
|
||||||
- Именование инодов через хранение их метаданных в etcd
|
- Именование инодов через хранение их метаданных в etcd
|
||||||
- Снапшоты и copy-on-write клоны
|
- Снапшоты и copy-on-write клоны
|
||||||
- Сглаживание производительности случайной записи в SSD+HDD конфигурациях
|
- Сглаживание производительности случайной записи в SSD+HDD конфигурациях
|
||||||
- Поддержка RDMA/RoCEv2 через libibverbs
|
|
||||||
- CSI-плагин для Kubernetes
|
|
||||||
- Базовая поддержка OpenStack: драйвер Cinder, патчи для Nova и libvirt
|
|
||||||
- Слияние снапшотов (vitastor-cli {snap-rm,flatten,merge})
|
|
||||||
- Консольный интерфейс для управления образами (vitastor-cli {ls,create,modify})
|
|
||||||
- Плагин для Proxmox
|
|
||||||
|
|
||||||
## Планы развития
|
## Планы развития
|
||||||
|
|
||||||
- Поддержка удаления снапшотов (слияния слоёв)
|
|
||||||
- Более корректные скрипты разметки дисков и автоматического запуска OSD
|
- Более корректные скрипты разметки дисков и автоматического запуска OSD
|
||||||
- Другие инструменты администрирования
|
- Другие инструменты администрирования
|
||||||
- Плагины для OpenNebula и других облачных систем
|
- Плагины для OpenStack, Kubernetes, OpenNebula, Proxmox и других облачных систем
|
||||||
- iSCSI-прокси
|
- iSCSI-прокси
|
||||||
- Более быстрое переключение при отказах
|
- Более быстрое переключение при отказах
|
||||||
- Фоновая проверка целостности без контрольных сумм (сверка реплик)
|
- Фоновая проверка целостности без контрольных сумм (сверка реплик)
|
||||||
- Контрольные суммы
|
- Контрольные суммы
|
||||||
- Поддержка SSD-кэширования (tiered storage)
|
- Поддержка SSD-кэширования (tiered storage)
|
||||||
- Поддержка NVDIMM
|
- Поддержка RDMA и NVDIMM
|
||||||
- Web-интерфейс
|
- Web-интерфейс
|
||||||
- Возможно, сжатие
|
- Возможно, сжатие
|
||||||
- Возможно, поддержка кэширования данных через системный page cache
|
- Возможно, поддержка кэширования данных через системный page cache
|
||||||
|
@ -375,7 +371,7 @@ Vitastor с однопоточной NBD прокси на том же стен
|
||||||
- Установите gcc и g++ 8.x или новее.
|
- Установите gcc и g++ 8.x или новее.
|
||||||
- Склонируйте данный репозиторий с подмодулями: `git clone https://yourcmc.ru/git/vitalif/vitastor/`.
|
- Склонируйте данный репозиторий с подмодулями: `git clone https://yourcmc.ru/git/vitalif/vitastor/`.
|
||||||
- Желательно пересобрать QEMU с патчем, который делает необязательным запуск через LD_PRELOAD.
|
- Желательно пересобрать QEMU с патчем, который делает необязательным запуск через LD_PRELOAD.
|
||||||
См `patches/qemu-*.*-vitastor.patch` - выберите версию, наиболее близкую вашей версии QEMU.
|
См `qemu-*.*-vitastor.patch` - выберите версию, наиболее близкую вашей версии QEMU.
|
||||||
- Установите QEMU 3.0 или новее, возьмите исходные коды установленного пакета, начните его пересборку,
|
- Установите QEMU 3.0 или новее, возьмите исходные коды установленного пакета, начните его пересборку,
|
||||||
через некоторое время остановите её и скопируйте следующие заголовки:
|
через некоторое время остановите её и скопируйте следующие заголовки:
|
||||||
- `<qemu>/include` → `<vitastor>/qemu/include`
|
- `<qemu>/include` → `<vitastor>/qemu/include`
|
||||||
|
@ -404,21 +400,12 @@ Vitastor с однопоточной NBD прокси на том же стен
|
||||||
в этом случае пострадает.
|
в этом случае пострадает.
|
||||||
- Быстрая сеть, минимум 10 гбит/с
|
- Быстрая сеть, минимум 10 гбит/с
|
||||||
- Для наилучшей производительности нужно отключить энергосбережение CPU: `cpupower idle-set -D 0 && cpupower frequency-set -g performance`.
|
- Для наилучшей производительности нужно отключить энергосбережение CPU: `cpupower idle-set -D 0 && cpupower frequency-set -g performance`.
|
||||||
- На хостах мониторов:
|
- Пропишите нужные вам значения вверху файлов `/usr/lib/vitastor/mon/make-units.sh` и `/usr/lib/vitastor/mon/make-osd.sh`.
|
||||||
- Пропишите нужные вам значения в файле `/usr/lib/vitastor/mon/make-units.sh`
|
- Создайте юниты systemd для etcd и мониторов: `/usr/lib/vitastor/mon/make-units.sh`
|
||||||
- Создайте юниты systemd для etcd и мониторов: `/usr/lib/vitastor/mon/make-units.sh`
|
- Создайте юниты для OSD: `/usr/lib/vitastor/mon/make-osd.sh /dev/disk/by-partuuid/XXX [/dev/disk/by-partuuid/YYY ...]`
|
||||||
- Пропишите etcd_address и osd_network в `/etc/vitastor/vitastor.conf`. Например:
|
- Вы можете поменять параметры OSD в юнитах systemd. Смысл некоторых параметров:
|
||||||
```
|
|
||||||
{
|
|
||||||
"etcd_address": ["10.200.1.10:2379","10.200.1.11:2379","10.200.1.12:2379"],
|
|
||||||
"osd_network": "10.200.1.0/24"
|
|
||||||
}
|
|
||||||
```
|
|
||||||
- Создайте юниты systemd для OSD: `/usr/lib/vitastor/make-osd.sh /dev/disk/by-partuuid/XXX [/dev/disk/by-partuuid/YYY ...]`
|
|
||||||
- Вы можете менять параметры OSD в юнитах systemd или в `vitastor.conf`. Смысл некоторых параметров:
|
|
||||||
- `disable_data_fsync 1` - отключает fsync, используется с SSD с конденсаторами.
|
- `disable_data_fsync 1` - отключает fsync, используется с SSD с конденсаторами.
|
||||||
- `immediate_commit all` - используется с SSD с конденсаторами.
|
- `immediate_commit all` - используется с SSD с конденсаторами.
|
||||||
Внимание: если установлено, также нужно установить его в то же значение в etcd в /vitastor/config/global
|
|
||||||
- `disable_device_lock 1` - отключает блокировку файла устройства, нужно, только если вы запускаете
|
- `disable_device_lock 1` - отключает блокировку файла устройства, нужно, только если вы запускаете
|
||||||
несколько OSD на одном блочном устройстве.
|
несколько OSD на одном блочном устройстве.
|
||||||
- `flusher_count 256` - "flusher" - микропоток, удаляющий старые данные из журнала.
|
- `flusher_count 256` - "flusher" - микропоток, удаляющий старые данные из журнала.
|
||||||
|
@ -503,10 +490,10 @@ qemu-system-x86_64 -enable-kvm -m 1024
|
||||||
|
|
||||||
### Удалить образ
|
### Удалить образ
|
||||||
|
|
||||||
Используйте утилиту vitastor-cli rm-data. Например:
|
Используйте утилиту vitastor-rm. Например:
|
||||||
|
|
||||||
```
|
```
|
||||||
vitastor-cli rm-data --etcd_address 10.115.0.10:2379/v3 --pool 1 --inode 1 --parallel_osds 16 --iodepth 32
|
vitastor-rm --etcd_address 10.115.0.10:2379/v3 --pool 1 --inode 1 --parallel_osds 16 --iodepth 32
|
||||||
```
|
```
|
||||||
|
|
||||||
### NBD
|
### NBD
|
||||||
|
@ -523,90 +510,6 @@ vitastor-nbd map --etcd_address 10.115.0.10:2379/v3 --image testimg
|
||||||
Для обращения по номеру инода, аналогично другим командам, можно использовать опции
|
Для обращения по номеру инода, аналогично другим командам, можно использовать опции
|
||||||
`--pool <POOL> --inode <INODE> --size <SIZE>` вместо `--image testimg`.
|
`--pool <POOL> --inode <INODE> --size <SIZE>` вместо `--image testimg`.
|
||||||
|
|
||||||
### Kubernetes
|
|
||||||
|
|
||||||
У Vitastor есть CSI-плагин для Kubernetes, поддерживающий RWO-тома.
|
|
||||||
|
|
||||||
Для установки возьмите манифесты из директории [csi/deploy/](csi/deploy/), поместите
|
|
||||||
вашу конфигурацию подключения к Vitastor в [csi/deploy/001-csi-config-map.yaml](001-csi-config-map.yaml),
|
|
||||||
настройте StorageClass в [csi/deploy/009-storage-class.yaml](009-storage-class.yaml)
|
|
||||||
и примените все `NNN-*.yaml` к вашей инсталляции Kubernetes.
|
|
||||||
|
|
||||||
```
|
|
||||||
for i in ./???-*.yaml; do kubectl apply -f $i; done
|
|
||||||
```
|
|
||||||
|
|
||||||
После этого вы сможете создавать PersistentVolume. Пример смотрите в файле [csi/deploy/example-pvc.yaml](csi/deploy/example-pvc.yaml).
|
|
||||||
|
|
||||||
### OpenStack
|
|
||||||
|
|
||||||
Чтобы подключить Vitastor к OpenStack:
|
|
||||||
|
|
||||||
- Установите пакеты vitastor-client, libvirt и QEMU из DEB или RPM репозитория Vitastor
|
|
||||||
- Примените патч `patches/nova-21.diff` или `patches/nova-23.diff` к вашей инсталляции Nova.
|
|
||||||
nova-21.diff подходит для Nova 21-22, nova-23.diff подходит для Nova 23-24.
|
|
||||||
- Скопируйте `patches/cinder-vitastor.py` в инсталляцию Cinder как `cinder/volume/drivers/vitastor.py`
|
|
||||||
- Создайте тип томов в cinder.conf (см. ниже)
|
|
||||||
- Обязательно заблокируйте доступ от виртуальных машин к сети Vitastor (OSD и etcd), т.к. Vitastor (пока) не поддерживает аутентификацию
|
|
||||||
- Перезапустите Cinder и Nova
|
|
||||||
|
|
||||||
Пример конфигурации Cinder:
|
|
||||||
|
|
||||||
```
|
|
||||||
[DEFAULT]
|
|
||||||
enabled_backends = lvmdriver-1, vitastor-testcluster
|
|
||||||
# ...
|
|
||||||
|
|
||||||
[vitastor-testcluster]
|
|
||||||
volume_driver = cinder.volume.drivers.vitastor.VitastorDriver
|
|
||||||
volume_backend_name = vitastor-testcluster
|
|
||||||
image_volume_cache_enabled = True
|
|
||||||
volume_clear = none
|
|
||||||
vitastor_etcd_address = 192.168.7.2:2379
|
|
||||||
vitastor_etcd_prefix =
|
|
||||||
vitastor_config_path = /etc/vitastor/vitastor.conf
|
|
||||||
vitastor_pool_id = 1
|
|
||||||
image_upload_use_cinder_backend = True
|
|
||||||
```
|
|
||||||
|
|
||||||
Чтобы помещать в Vitastor Glance-образы, нужно использовать
|
|
||||||
[https://docs.openstack.org/cinder/pike/admin/blockstorage-volume-backed-image.html](образы на основе томов Cinder),
|
|
||||||
однако, поддержка этой функции ещё не проверялась.
|
|
||||||
|
|
||||||
### Proxmox
|
|
||||||
|
|
||||||
Чтобы подключить Vitastor к Proxmox Virtual Environment (поддерживаются версии 6.4 и 7.1):
|
|
||||||
|
|
||||||
- Добавьте соответствующий Debian-репозиторий Vitastor в sources.list на хостах Proxmox
|
|
||||||
(buster для 6.4, bullseye для 7.1)
|
|
||||||
- Установите пакеты vitastor-client, pve-qemu-kvm, pve-storage-vitastor (* или см. сноску) из репозитория Vitastor
|
|
||||||
- Определите тип хранилища в `/etc/pve/storage.cfg` (см. ниже)
|
|
||||||
- Обязательно заблокируйте доступ от виртуальных машин к сети Vitastor (OSD и etcd), т.к. Vitastor (пока) не поддерживает аутентификацию
|
|
||||||
- Перезапустите демон Proxmox: `systemctl restart pvedaemon`
|
|
||||||
|
|
||||||
Пример `/etc/pve/storage.cfg` (единственная обязательная опция - vitastor_pool, все остальные
|
|
||||||
перечислены внизу для понимания значений по умолчанию):
|
|
||||||
|
|
||||||
```
|
|
||||||
vitastor: vitastor
|
|
||||||
# Пул, в который будут помещаться образы дисков
|
|
||||||
vitastor_pool testpool
|
|
||||||
# Путь к файлу конфигурации
|
|
||||||
vitastor_config_path /etc/vitastor/vitastor.conf
|
|
||||||
# Адрес(а) etcd, нужны, только если не указаны в vitastor.conf
|
|
||||||
vitastor_etcd_address 192.168.7.2:2379/v3
|
|
||||||
# Префикс ключей метаданных в etcd
|
|
||||||
vitastor_etcd_prefix /vitastor
|
|
||||||
# Префикс имён образов
|
|
||||||
vitastor_prefix pve/
|
|
||||||
# Монтировать образы через NBD прокси, через ядро (нужно только для контейнеров)
|
|
||||||
vitastor_nbd 0
|
|
||||||
```
|
|
||||||
|
|
||||||
\* Примечание: вместо установки пакета pve-storage-vitastor вы можете вручную скопировать файл
|
|
||||||
[patches/PVE_VitastorPlugin.pm](patches/PVE_VitastorPlugin.pm) на хосты Proxmox как
|
|
||||||
`/usr/share/perl5/PVE/Storage/Custom/VitastorPlugin.pm`.
|
|
||||||
|
|
||||||
## Известные проблемы
|
## Известные проблемы
|
||||||
|
|
||||||
- Запросы удаления объектов могут в данный момент приводить к "неполным" объектам в EC-пулах,
|
- Запросы удаления объектов могут в данный момент приводить к "неполным" объектам в EC-пулах,
|
||||||
|
|
120
README.md
120
README.md
|
@ -16,6 +16,7 @@ with configurable redundancy (replication or erasure codes/XOR).
|
||||||
Vitastor is currently a pre-release, a lot of features are missing and you can still expect
|
Vitastor is currently a pre-release, a lot of features are missing and you can still expect
|
||||||
breaking changes in the future. However, the following is implemented:
|
breaking changes in the future. However, the following is implemented:
|
||||||
|
|
||||||
|
0.5.x (stable):
|
||||||
- Basic part: highly-available block storage with symmetric clustering and no SPOF
|
- Basic part: highly-available block storage with symmetric clustering and no SPOF
|
||||||
- Performance ;-D
|
- Performance ;-D
|
||||||
- Multiple redundancy schemes: Replication, XOR n+1, Reed-Solomon erasure codes
|
- Multiple redundancy schemes: Replication, XOR n+1, Reed-Solomon erasure codes
|
||||||
|
@ -34,31 +35,26 @@ breaking changes in the future. However, the following is implemented:
|
||||||
- QEMU driver (built out-of-tree)
|
- QEMU driver (built out-of-tree)
|
||||||
- Loadable fio engine for benchmarks (also built out-of-tree)
|
- Loadable fio engine for benchmarks (also built out-of-tree)
|
||||||
- NBD proxy for kernel mounts
|
- NBD proxy for kernel mounts
|
||||||
- Inode removal tool (vitastor-cli rm-data)
|
- Inode removal tool (vitastor-rm)
|
||||||
- Packaging for Debian and CentOS
|
- Packaging for Debian and CentOS
|
||||||
|
|
||||||
|
0.6.x (master):
|
||||||
- Per-inode I/O and space usage statistics
|
- Per-inode I/O and space usage statistics
|
||||||
- Inode metadata storage in etcd
|
- Inode metadata storage in etcd
|
||||||
- Snapshots and copy-on-write image clones
|
- Snapshots and copy-on-write image clones
|
||||||
- Write throttling to smooth random write workloads in SSD+HDD configurations
|
- Write throttling to smooth random write workloads in SSD+HDD configurations
|
||||||
- RDMA/RoCEv2 support via libibverbs
|
|
||||||
- CSI plugin for Kubernetes
|
|
||||||
- Basic OpenStack support: Cinder driver, Nova and libvirt patches
|
|
||||||
- Snapshot merge tool (vitastor-cli {snap-rm,flatten,merge})
|
|
||||||
- Image management CLI (vitastor-cli {ls,create,modify})
|
|
||||||
- Proxmox storage plugin
|
|
||||||
|
|
||||||
## Roadmap
|
## Roadmap
|
||||||
|
|
||||||
- Snapshot deletion (layer merge) support
|
|
||||||
- Better OSD creation and auto-start tools
|
- Better OSD creation and auto-start tools
|
||||||
- Other administrative tools
|
- Other administrative tools
|
||||||
- Plugins for OpenNebula, Proxmox and other cloud systems
|
- Plugins for OpenStack, Kubernetes, OpenNebula, Proxmox and other cloud systems
|
||||||
- iSCSI proxy
|
- iSCSI proxy
|
||||||
- Faster failover
|
- Faster failover
|
||||||
- Scrubbing without checksums (verification of replicas)
|
- Scrubbing without checksums (verification of replicas)
|
||||||
- Checksums
|
- Checksums
|
||||||
- Tiered storage
|
- Tiered storage
|
||||||
- NVDIMM support
|
- RDMA and NVDIMM support
|
||||||
- Web GUI
|
- Web GUI
|
||||||
- Compression (possibly)
|
- Compression (possibly)
|
||||||
- Read caching using system page cache (possibly)
|
- Read caching using system page cache (possibly)
|
||||||
|
@ -343,7 +339,7 @@ Vitastor with single-thread NBD on the same hardware:
|
||||||
* For QEMU 2.0+: `<qemu>/qapi-types.h` → `<vitastor>/qemu/b/qemu/qapi-types.h`
|
* For QEMU 2.0+: `<qemu>/qapi-types.h` → `<vitastor>/qemu/b/qemu/qapi-types.h`
|
||||||
- `config-host.h` and `qapi` are required because they contain generated headers
|
- `config-host.h` and `qapi` are required because they contain generated headers
|
||||||
- You can also rebuild QEMU with a patch that makes LD_PRELOAD unnecessary to load vitastor driver.
|
- You can also rebuild QEMU with a patch that makes LD_PRELOAD unnecessary to load vitastor driver.
|
||||||
See `patches/qemu-*.*-vitastor.patch`.
|
See `qemu-*.*-vitastor.patch`.
|
||||||
- Install fio 3.7 or later, get its source and symlink it into `<vitastor>/fio`.
|
- Install fio 3.7 or later, get its source and symlink it into `<vitastor>/fio`.
|
||||||
- Build & install Vitastor with `mkdir build && cd build && cmake .. && make -j8 && make install`.
|
- Build & install Vitastor with `mkdir build && cd build && cmake .. && make -j8 && make install`.
|
||||||
Pay attention to the `QEMU_PLUGINDIR` cmake option - it must be set to `qemu-kvm` on RHEL.
|
Pay attention to the `QEMU_PLUGINDIR` cmake option - it must be set to `qemu-kvm` on RHEL.
|
||||||
|
@ -357,21 +353,13 @@ and calculate disk offsets almost by hand. This will be fixed in near future.
|
||||||
with lazy fsync, but prepare for inferior single-thread latency.
|
with lazy fsync, but prepare for inferior single-thread latency.
|
||||||
- Get a fast network (at least 10 Gbit/s).
|
- Get a fast network (at least 10 Gbit/s).
|
||||||
- Disable CPU powersaving: `cpupower idle-set -D 0 && cpupower frequency-set -g performance`.
|
- Disable CPU powersaving: `cpupower idle-set -D 0 && cpupower frequency-set -g performance`.
|
||||||
- On the monitor hosts:
|
- Check `/usr/lib/vitastor/mon/make-units.sh` and `/usr/lib/vitastor/mon/make-osd.sh` and
|
||||||
- Edit variables at the top of `/usr/lib/vitastor/mon/make-units.sh` to desired values.
|
put desired values into the variables at the top of these files.
|
||||||
- Create systemd units for the monitor and etcd: `/usr/lib/vitastor/mon/make-units.sh`
|
- Create systemd units for the monitor and etcd: `/usr/lib/vitastor/mon/make-units.sh`
|
||||||
- Put etcd_address and osd_network into `/etc/vitastor/vitastor.conf`. Example:
|
|
||||||
```
|
|
||||||
{
|
|
||||||
"etcd_address": ["10.200.1.10:2379","10.200.1.11:2379","10.200.1.12:2379"],
|
|
||||||
"osd_network": "10.200.1.0/24"
|
|
||||||
}
|
|
||||||
```
|
|
||||||
- Create systemd units for your OSDs: `/usr/lib/vitastor/mon/make-osd.sh /dev/disk/by-partuuid/XXX [/dev/disk/by-partuuid/YYY ...]`
|
- Create systemd units for your OSDs: `/usr/lib/vitastor/mon/make-osd.sh /dev/disk/by-partuuid/XXX [/dev/disk/by-partuuid/YYY ...]`
|
||||||
- You can change OSD configuration in units or in `vitastor.conf`. Notable configuration variables:
|
- You can edit the units and change OSD configuration. Notable configuration variables:
|
||||||
- `disable_data_fsync 1` - only safe with server-grade drives with capacitors.
|
- `disable_data_fsync 1` - only safe with server-grade drives with capacitors.
|
||||||
- `immediate_commit all` - use this if all your drives are server-grade.
|
- `immediate_commit all` - use this if all your drives are server-grade.
|
||||||
If all OSDs have it set to all then you should also put the same value in etcd into /vitastor/config/global
|
|
||||||
- `disable_device_lock 1` - only required if you run multiple OSDs on one block device.
|
- `disable_device_lock 1` - only required if you run multiple OSDs on one block device.
|
||||||
- `flusher_count 256` - flusher is a micro-thread that removes old data from the journal.
|
- `flusher_count 256` - flusher is a micro-thread that removes old data from the journal.
|
||||||
You don't have to worry about this parameter anymore, 256 is enough.
|
You don't have to worry about this parameter anymore, 256 is enough.
|
||||||
|
@ -454,10 +442,10 @@ just like in qemu-img.
|
||||||
|
|
||||||
### Remove inode
|
### Remove inode
|
||||||
|
|
||||||
Use vitastor-rm / vitastor-cli rm-data. For example:
|
Use vitastor-rm. For example:
|
||||||
|
|
||||||
```
|
```
|
||||||
vitastor-cli rm-data --etcd_address 10.115.0.10:2379/v3 --pool 1 --inode 1 --parallel_osds 16 --iodepth 32
|
vitastor-rm --etcd_address 10.115.0.10:2379/v3 --pool 1 --inode 1 --parallel_osds 16 --iodepth 32
|
||||||
```
|
```
|
||||||
|
|
||||||
### NBD
|
### NBD
|
||||||
|
@ -472,88 +460,6 @@ It will output the device name, like /dev/nbd0 which you can then format and mou
|
||||||
|
|
||||||
Again, you can use `--pool <POOL> --inode <INODE> --size <SIZE>` insteaf of `--image <IMAGE>` if you want.
|
Again, you can use `--pool <POOL> --inode <INODE> --size <SIZE>` insteaf of `--image <IMAGE>` if you want.
|
||||||
|
|
||||||
### Kubernetes
|
|
||||||
|
|
||||||
Vitastor has a CSI plugin for Kubernetes which supports RWO volumes.
|
|
||||||
|
|
||||||
To deploy it, take manifests from [csi/deploy/](csi/deploy/) directory, put your
|
|
||||||
Vitastor configuration in [csi/deploy/001-csi-config-map.yaml](001-csi-config-map.yaml),
|
|
||||||
configure storage class in [csi/deploy/009-storage-class.yaml](009-storage-class.yaml)
|
|
||||||
and apply all `NNN-*.yaml` manifests to your Kubernetes installation:
|
|
||||||
|
|
||||||
```
|
|
||||||
for i in ./???-*.yaml; do kubectl apply -f $i; done
|
|
||||||
```
|
|
||||||
|
|
||||||
After that you'll be able to create PersistentVolumes. See example in [csi/deploy/example-pvc.yaml](csi/deploy/example-pvc.yaml).
|
|
||||||
|
|
||||||
### OpenStack
|
|
||||||
|
|
||||||
To enable Vitastor support in an OpenStack installation:
|
|
||||||
|
|
||||||
- Install vitastor-client, patched QEMU and libvirt packages from Vitastor DEB or RPM repository
|
|
||||||
- Use `patches/nova-21.diff` or `patches/nova-23.diff` to patch your Nova installation.
|
|
||||||
Patch 21 fits Nova 21-22, patch 23 fits Nova 23-24.
|
|
||||||
- Install `patches/cinder-vitastor.py` as `..../cinder/volume/drivers/vitastor.py`
|
|
||||||
- Define a volume type in cinder.conf (see below)
|
|
||||||
- Block network access from VMs to Vitastor network (to OSDs and etcd), because Vitastor doesn't support authentication (yet)
|
|
||||||
- Restart Cinder and Nova
|
|
||||||
|
|
||||||
Cinder volume type configuration example:
|
|
||||||
|
|
||||||
```
|
|
||||||
[DEFAULT]
|
|
||||||
enabled_backends = lvmdriver-1, vitastor-testcluster
|
|
||||||
# ...
|
|
||||||
|
|
||||||
[vitastor-testcluster]
|
|
||||||
volume_driver = cinder.volume.drivers.vitastor.VitastorDriver
|
|
||||||
volume_backend_name = vitastor-testcluster
|
|
||||||
image_volume_cache_enabled = True
|
|
||||||
volume_clear = none
|
|
||||||
vitastor_etcd_address = 192.168.7.2:2379
|
|
||||||
vitastor_etcd_prefix =
|
|
||||||
vitastor_config_path = /etc/vitastor/vitastor.conf
|
|
||||||
vitastor_pool_id = 1
|
|
||||||
image_upload_use_cinder_backend = True
|
|
||||||
```
|
|
||||||
|
|
||||||
To put Glance images in Vitastor, use [https://docs.openstack.org/cinder/pike/admin/blockstorage-volume-backed-image.html](volume-backed images),
|
|
||||||
although the support has not been verified yet.
|
|
||||||
|
|
||||||
### Proxmox
|
|
||||||
|
|
||||||
To enable Vitastor support in Proxmox Virtual Environment (6.4 and 7.1 are supported):
|
|
||||||
|
|
||||||
- Add the corresponding Vitastor Debian repository into sources.list on Proxmox hosts
|
|
||||||
(buster for 6.4, bullseye for 7.1)
|
|
||||||
- Install vitastor-client, pve-qemu-kvm, pve-storage-vitastor (* or see note) packages from Vitastor repository
|
|
||||||
- Define storage in `/etc/pve/storage.cfg` (see below)
|
|
||||||
- Block network access from VMs to Vitastor network (to OSDs and etcd), because Vitastor doesn't support authentication (yet)
|
|
||||||
- Restart pvedaemon: `systemctl restart pvedaemon`
|
|
||||||
|
|
||||||
`/etc/pve/storage.cfg` example (the only required option is vitastor_pool, all others
|
|
||||||
are listed below with their default values):
|
|
||||||
|
|
||||||
```
|
|
||||||
vitastor: vitastor
|
|
||||||
# pool to put new images into
|
|
||||||
vitastor_pool testpool
|
|
||||||
# path to the configuration file
|
|
||||||
vitastor_config_path /etc/vitastor/vitastor.conf
|
|
||||||
# etcd address(es), required only if missing in the configuration file
|
|
||||||
vitastor_etcd_address 192.168.7.2:2379/v3
|
|
||||||
# prefix for keys in etcd
|
|
||||||
vitastor_etcd_prefix /vitastor
|
|
||||||
# prefix for images
|
|
||||||
vitastor_prefix pve/
|
|
||||||
# use NBD mounter (only required for containers)
|
|
||||||
vitastor_nbd 0
|
|
||||||
```
|
|
||||||
|
|
||||||
\* Note: you can also manually copy [patches/PVE_VitastorPlugin.pm](patches/PVE_VitastorPlugin.pm) to Proxmox hosts
|
|
||||||
as `/usr/share/perl5/PVE/Storage/Custom/VitastorPlugin.pm` instead of installing pve-storage-vitastor.
|
|
||||||
|
|
||||||
## Known Problems
|
## Known Problems
|
||||||
|
|
||||||
- Object deletion requests may currently lead to 'incomplete' objects in EC pools
|
- Object deletion requests may currently lead to 'incomplete' objects in EC pools
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
vitastor-csi
|
|
||||||
Dockerfile
|
|
|
@ -1,32 +0,0 @@
|
||||||
# Compile stage
|
|
||||||
FROM golang:buster AS build
|
|
||||||
|
|
||||||
ADD go.sum go.mod /app/
|
|
||||||
RUN cd /app; CGO_ENABLED=1 GOOS=linux GOARCH=amd64 go mod download -x
|
|
||||||
ADD . /app
|
|
||||||
RUN perl -i -e '$/ = undef; while(<>) { s/\n\s*(\{\s*\n)/$1\n/g; s/\}(\s*\n\s*)else\b/$1} else/g; print; }' `find /app -name '*.go'`
|
|
||||||
RUN cd /app; CGO_ENABLED=1 GOOS=linux GOARCH=amd64 go build -o vitastor-csi
|
|
||||||
|
|
||||||
# Final stage
|
|
||||||
FROM debian:buster
|
|
||||||
|
|
||||||
LABEL maintainers="Vitaliy Filippov <vitalif@yourcmc.ru>"
|
|
||||||
LABEL description="Vitastor CSI Driver"
|
|
||||||
|
|
||||||
ENV NODE_ID=""
|
|
||||||
ENV CSI_ENDPOINT=""
|
|
||||||
|
|
||||||
RUN apt-get update && \
|
|
||||||
apt-get install -y wget && \
|
|
||||||
wget -q -O /etc/apt/trusted.gpg.d/vitastor.gpg https://vitastor.io/debian/pubkey.gpg && \
|
|
||||||
(echo deb http://vitastor.io/debian buster main > /etc/apt/sources.list.d/vitastor.list) && \
|
|
||||||
(echo deb http://deb.debian.org/debian buster-backports main > /etc/apt/sources.list.d/backports.list) && \
|
|
||||||
(echo "APT::Install-Recommends false;" > /etc/apt/apt.conf) && \
|
|
||||||
apt-get update && \
|
|
||||||
apt-get install -y e2fsprogs xfsprogs vitastor kmod && \
|
|
||||||
apt-get clean && \
|
|
||||||
(echo options nbd nbds_max=128 > /etc/modprobe.d/nbd.conf)
|
|
||||||
|
|
||||||
COPY --from=build /app/vitastor-csi /bin/
|
|
||||||
|
|
||||||
ENTRYPOINT ["/bin/vitastor-csi"]
|
|
|
@ -1,9 +0,0 @@
|
||||||
VERSION ?= v0.6.10
|
|
||||||
|
|
||||||
all: build push
|
|
||||||
|
|
||||||
build:
|
|
||||||
@docker build --rm -t vitalif/vitastor-csi:$(VERSION) .
|
|
||||||
|
|
||||||
push:
|
|
||||||
@docker push vitalif/vitastor-csi:$(VERSION)
|
|
|
@ -1,5 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: Namespace
|
|
||||||
metadata:
|
|
||||||
name: vitastor-system
|
|
|
@ -1,9 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ConfigMap
|
|
||||||
data:
|
|
||||||
vitastor.conf: |-
|
|
||||||
{"etcd_address":"http://192.168.7.2:2379","etcd_prefix":"/vitastor"}
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-config
|
|
|
@ -1,37 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ServiceAccount
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-nodeplugin
|
|
||||||
---
|
|
||||||
kind: ClusterRole
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-nodeplugin
|
|
||||||
rules:
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["nodes"]
|
|
||||||
verbs: ["get"]
|
|
||||||
# allow to read Vault Token and connection options from the Tenants namespace
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["secrets"]
|
|
||||||
verbs: ["get"]
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["configmaps"]
|
|
||||||
verbs: ["get"]
|
|
||||||
---
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-nodeplugin
|
|
||||||
subjects:
|
|
||||||
- kind: ServiceAccount
|
|
||||||
name: vitastor-csi-nodeplugin
|
|
||||||
namespace: vitastor-system
|
|
||||||
roleRef:
|
|
||||||
kind: ClusterRole
|
|
||||||
name: vitastor-csi-nodeplugin
|
|
||||||
apiGroup: rbac.authorization.k8s.io
|
|
|
@ -1,72 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: policy/v1beta1
|
|
||||||
kind: PodSecurityPolicy
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-nodeplugin-psp
|
|
||||||
spec:
|
|
||||||
allowPrivilegeEscalation: true
|
|
||||||
allowedCapabilities:
|
|
||||||
- 'SYS_ADMIN'
|
|
||||||
fsGroup:
|
|
||||||
rule: RunAsAny
|
|
||||||
privileged: true
|
|
||||||
hostNetwork: true
|
|
||||||
hostPID: true
|
|
||||||
runAsUser:
|
|
||||||
rule: RunAsAny
|
|
||||||
seLinux:
|
|
||||||
rule: RunAsAny
|
|
||||||
supplementalGroups:
|
|
||||||
rule: RunAsAny
|
|
||||||
volumes:
|
|
||||||
- 'configMap'
|
|
||||||
- 'emptyDir'
|
|
||||||
- 'projected'
|
|
||||||
- 'secret'
|
|
||||||
- 'downwardAPI'
|
|
||||||
- 'hostPath'
|
|
||||||
allowedHostPaths:
|
|
||||||
- pathPrefix: '/dev'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/run/mount'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/sys'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/lib/modules'
|
|
||||||
readOnly: true
|
|
||||||
- pathPrefix: '/var/lib/kubelet/pods'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/var/lib/kubelet/plugins/csi.vitastor.io'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/var/lib/kubelet/plugins_registry'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/var/lib/kubelet/plugins'
|
|
||||||
readOnly: false
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: Role
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-nodeplugin-psp
|
|
||||||
rules:
|
|
||||||
- apiGroups: ['policy']
|
|
||||||
resources: ['podsecuritypolicies']
|
|
||||||
verbs: ['use']
|
|
||||||
resourceNames: ['vitastor-csi-nodeplugin-psp']
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: RoleBinding
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-nodeplugin-psp
|
|
||||||
subjects:
|
|
||||||
- kind: ServiceAccount
|
|
||||||
name: vitastor-csi-nodeplugin
|
|
||||||
namespace: vitastor-system
|
|
||||||
roleRef:
|
|
||||||
kind: Role
|
|
||||||
name: vitastor-csi-nodeplugin-psp
|
|
||||||
apiGroup: rbac.authorization.k8s.io
|
|
|
@ -1,140 +0,0 @@
|
||||||
---
|
|
||||||
kind: DaemonSet
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: csi-vitastor
|
|
||||||
spec:
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app: csi-vitastor
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
labels:
|
|
||||||
app: csi-vitastor
|
|
||||||
spec:
|
|
||||||
serviceAccountName: vitastor-csi-nodeplugin
|
|
||||||
hostNetwork: true
|
|
||||||
hostPID: true
|
|
||||||
priorityClassName: system-node-critical
|
|
||||||
# to use e.g. Rook orchestrated cluster, and mons' FQDN is
|
|
||||||
# resolved through k8s service, set dns policy to cluster first
|
|
||||||
dnsPolicy: ClusterFirstWithHostNet
|
|
||||||
containers:
|
|
||||||
- name: driver-registrar
|
|
||||||
# This is necessary only for systems with SELinux, where
|
|
||||||
# non-privileged sidecar containers cannot access unix domain socket
|
|
||||||
# created by privileged CSI driver container.
|
|
||||||
securityContext:
|
|
||||||
privileged: true
|
|
||||||
image: k8s.gcr.io/sig-storage/csi-node-driver-registrar:v2.2.0
|
|
||||||
args:
|
|
||||||
- "--v=5"
|
|
||||||
- "--csi-address=/csi/csi.sock"
|
|
||||||
- "--kubelet-registration-path=/var/lib/kubelet/plugins/csi.vitastor.io/csi.sock"
|
|
||||||
env:
|
|
||||||
- name: KUBE_NODE_NAME
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: spec.nodeName
|
|
||||||
volumeMounts:
|
|
||||||
- name: socket-dir
|
|
||||||
mountPath: /csi
|
|
||||||
- name: registration-dir
|
|
||||||
mountPath: /registration
|
|
||||||
- name: csi-vitastor
|
|
||||||
securityContext:
|
|
||||||
privileged: true
|
|
||||||
capabilities:
|
|
||||||
add: ["SYS_ADMIN"]
|
|
||||||
allowPrivilegeEscalation: true
|
|
||||||
image: vitalif/vitastor-csi:v0.6.10
|
|
||||||
args:
|
|
||||||
- "--node=$(NODE_ID)"
|
|
||||||
- "--endpoint=$(CSI_ENDPOINT)"
|
|
||||||
env:
|
|
||||||
- name: NODE_ID
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: spec.nodeName
|
|
||||||
- name: CSI_ENDPOINT
|
|
||||||
value: unix:///csi/csi.sock
|
|
||||||
imagePullPolicy: "IfNotPresent"
|
|
||||||
ports:
|
|
||||||
- containerPort: 9898
|
|
||||||
name: healthz
|
|
||||||
protocol: TCP
|
|
||||||
livenessProbe:
|
|
||||||
failureThreshold: 5
|
|
||||||
httpGet:
|
|
||||||
path: /healthz
|
|
||||||
port: healthz
|
|
||||||
initialDelaySeconds: 10
|
|
||||||
timeoutSeconds: 3
|
|
||||||
periodSeconds: 2
|
|
||||||
volumeMounts:
|
|
||||||
- name: socket-dir
|
|
||||||
mountPath: /csi
|
|
||||||
- mountPath: /dev
|
|
||||||
name: host-dev
|
|
||||||
- mountPath: /sys
|
|
||||||
name: host-sys
|
|
||||||
- mountPath: /run/mount
|
|
||||||
name: host-mount
|
|
||||||
- mountPath: /lib/modules
|
|
||||||
name: lib-modules
|
|
||||||
readOnly: true
|
|
||||||
- name: vitastor-config
|
|
||||||
mountPath: /etc/vitastor
|
|
||||||
- name: plugin-dir
|
|
||||||
mountPath: /var/lib/kubelet/plugins
|
|
||||||
mountPropagation: "Bidirectional"
|
|
||||||
- name: mountpoint-dir
|
|
||||||
mountPath: /var/lib/kubelet/pods
|
|
||||||
mountPropagation: "Bidirectional"
|
|
||||||
- name: liveness-probe
|
|
||||||
securityContext:
|
|
||||||
privileged: true
|
|
||||||
image: quay.io/k8scsi/livenessprobe:v1.1.0
|
|
||||||
args:
|
|
||||||
- "--csi-address=$(CSI_ENDPOINT)"
|
|
||||||
- "--health-port=9898"
|
|
||||||
env:
|
|
||||||
- name: CSI_ENDPOINT
|
|
||||||
value: unix://csi/csi.sock
|
|
||||||
volumeMounts:
|
|
||||||
- mountPath: /csi
|
|
||||||
name: socket-dir
|
|
||||||
volumes:
|
|
||||||
- name: socket-dir
|
|
||||||
hostPath:
|
|
||||||
path: /var/lib/kubelet/plugins/csi.vitastor.io
|
|
||||||
type: DirectoryOrCreate
|
|
||||||
- name: plugin-dir
|
|
||||||
hostPath:
|
|
||||||
path: /var/lib/kubelet/plugins
|
|
||||||
type: Directory
|
|
||||||
- name: mountpoint-dir
|
|
||||||
hostPath:
|
|
||||||
path: /var/lib/kubelet/pods
|
|
||||||
type: DirectoryOrCreate
|
|
||||||
- name: registration-dir
|
|
||||||
hostPath:
|
|
||||||
path: /var/lib/kubelet/plugins_registry/
|
|
||||||
type: Directory
|
|
||||||
- name: host-dev
|
|
||||||
hostPath:
|
|
||||||
path: /dev
|
|
||||||
- name: host-sys
|
|
||||||
hostPath:
|
|
||||||
path: /sys
|
|
||||||
- name: host-mount
|
|
||||||
hostPath:
|
|
||||||
path: /run/mount
|
|
||||||
- name: lib-modules
|
|
||||||
hostPath:
|
|
||||||
path: /lib/modules
|
|
||||||
- name: vitastor-config
|
|
||||||
configMap:
|
|
||||||
name: vitastor-config
|
|
|
@ -1,102 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: ServiceAccount
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-provisioner
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: ClusterRole
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-external-provisioner-runner
|
|
||||||
rules:
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["nodes"]
|
|
||||||
verbs: ["get", "list", "watch"]
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["secrets"]
|
|
||||||
verbs: ["get", "list", "watch"]
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["events"]
|
|
||||||
verbs: ["list", "watch", "create", "update", "patch"]
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["persistentvolumes"]
|
|
||||||
verbs: ["get", "list", "watch", "create", "update", "delete", "patch"]
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["persistentvolumeclaims"]
|
|
||||||
verbs: ["get", "list", "watch", "update"]
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["persistentvolumeclaims/status"]
|
|
||||||
verbs: ["update", "patch"]
|
|
||||||
- apiGroups: ["storage.k8s.io"]
|
|
||||||
resources: ["storageclasses"]
|
|
||||||
verbs: ["get", "list", "watch"]
|
|
||||||
- apiGroups: ["snapshot.storage.k8s.io"]
|
|
||||||
resources: ["volumesnapshots"]
|
|
||||||
verbs: ["get", "list"]
|
|
||||||
- apiGroups: ["snapshot.storage.k8s.io"]
|
|
||||||
resources: ["volumesnapshotcontents"]
|
|
||||||
verbs: ["create", "get", "list", "watch", "update", "delete"]
|
|
||||||
- apiGroups: ["snapshot.storage.k8s.io"]
|
|
||||||
resources: ["volumesnapshotclasses"]
|
|
||||||
verbs: ["get", "list", "watch"]
|
|
||||||
- apiGroups: ["storage.k8s.io"]
|
|
||||||
resources: ["volumeattachments"]
|
|
||||||
verbs: ["get", "list", "watch", "update", "patch"]
|
|
||||||
- apiGroups: ["storage.k8s.io"]
|
|
||||||
resources: ["volumeattachments/status"]
|
|
||||||
verbs: ["patch"]
|
|
||||||
- apiGroups: ["storage.k8s.io"]
|
|
||||||
resources: ["csinodes"]
|
|
||||||
verbs: ["get", "list", "watch"]
|
|
||||||
- apiGroups: ["snapshot.storage.k8s.io"]
|
|
||||||
resources: ["volumesnapshotcontents/status"]
|
|
||||||
verbs: ["update"]
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["configmaps"]
|
|
||||||
verbs: ["get"]
|
|
||||||
---
|
|
||||||
kind: ClusterRoleBinding
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-provisioner-role
|
|
||||||
subjects:
|
|
||||||
- kind: ServiceAccount
|
|
||||||
name: vitastor-csi-provisioner
|
|
||||||
namespace: vitastor-system
|
|
||||||
roleRef:
|
|
||||||
kind: ClusterRole
|
|
||||||
name: vitastor-external-provisioner-runner
|
|
||||||
apiGroup: rbac.authorization.k8s.io
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: Role
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-external-provisioner-cfg
|
|
||||||
rules:
|
|
||||||
- apiGroups: [""]
|
|
||||||
resources: ["configmaps"]
|
|
||||||
verbs: ["get", "list", "watch", "create", "update", "delete"]
|
|
||||||
- apiGroups: ["coordination.k8s.io"]
|
|
||||||
resources: ["leases"]
|
|
||||||
verbs: ["get", "watch", "list", "delete", "update", "create"]
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: RoleBinding
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
name: vitastor-csi-provisioner-role-cfg
|
|
||||||
namespace: vitastor-system
|
|
||||||
subjects:
|
|
||||||
- kind: ServiceAccount
|
|
||||||
name: vitastor-csi-provisioner
|
|
||||||
namespace: vitastor-system
|
|
||||||
roleRef:
|
|
||||||
kind: Role
|
|
||||||
name: vitastor-external-provisioner-cfg
|
|
||||||
apiGroup: rbac.authorization.k8s.io
|
|
|
@ -1,60 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: policy/v1beta1
|
|
||||||
kind: PodSecurityPolicy
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-provisioner-psp
|
|
||||||
spec:
|
|
||||||
allowPrivilegeEscalation: true
|
|
||||||
allowedCapabilities:
|
|
||||||
- 'SYS_ADMIN'
|
|
||||||
fsGroup:
|
|
||||||
rule: RunAsAny
|
|
||||||
privileged: true
|
|
||||||
runAsUser:
|
|
||||||
rule: RunAsAny
|
|
||||||
seLinux:
|
|
||||||
rule: RunAsAny
|
|
||||||
supplementalGroups:
|
|
||||||
rule: RunAsAny
|
|
||||||
volumes:
|
|
||||||
- 'configMap'
|
|
||||||
- 'emptyDir'
|
|
||||||
- 'projected'
|
|
||||||
- 'secret'
|
|
||||||
- 'downwardAPI'
|
|
||||||
- 'hostPath'
|
|
||||||
allowedHostPaths:
|
|
||||||
- pathPrefix: '/dev'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/sys'
|
|
||||||
readOnly: false
|
|
||||||
- pathPrefix: '/lib/modules'
|
|
||||||
readOnly: true
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: Role
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor-csi-provisioner-psp
|
|
||||||
rules:
|
|
||||||
- apiGroups: ['policy']
|
|
||||||
resources: ['podsecuritypolicies']
|
|
||||||
verbs: ['use']
|
|
||||||
resourceNames: ['vitastor-csi-provisioner-psp']
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: RoleBinding
|
|
||||||
apiVersion: rbac.authorization.k8s.io/v1
|
|
||||||
metadata:
|
|
||||||
name: vitastor-csi-provisioner-psp
|
|
||||||
namespace: vitastor-system
|
|
||||||
subjects:
|
|
||||||
- kind: ServiceAccount
|
|
||||||
name: vitastor-csi-provisioner
|
|
||||||
namespace: vitastor-system
|
|
||||||
roleRef:
|
|
||||||
kind: Role
|
|
||||||
name: vitastor-csi-provisioner-psp
|
|
||||||
apiGroup: rbac.authorization.k8s.io
|
|
|
@ -1,159 +0,0 @@
|
||||||
---
|
|
||||||
kind: Service
|
|
||||||
apiVersion: v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: csi-vitastor-provisioner
|
|
||||||
labels:
|
|
||||||
app: csi-metrics
|
|
||||||
spec:
|
|
||||||
selector:
|
|
||||||
app: csi-vitastor-provisioner
|
|
||||||
ports:
|
|
||||||
- name: http-metrics
|
|
||||||
port: 8080
|
|
||||||
protocol: TCP
|
|
||||||
targetPort: 8680
|
|
||||||
|
|
||||||
---
|
|
||||||
kind: Deployment
|
|
||||||
apiVersion: apps/v1
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: csi-vitastor-provisioner
|
|
||||||
spec:
|
|
||||||
replicas: 3
|
|
||||||
selector:
|
|
||||||
matchLabels:
|
|
||||||
app: csi-vitastor-provisioner
|
|
||||||
template:
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
labels:
|
|
||||||
app: csi-vitastor-provisioner
|
|
||||||
spec:
|
|
||||||
affinity:
|
|
||||||
podAntiAffinity:
|
|
||||||
requiredDuringSchedulingIgnoredDuringExecution:
|
|
||||||
- labelSelector:
|
|
||||||
matchExpressions:
|
|
||||||
- key: app
|
|
||||||
operator: In
|
|
||||||
values:
|
|
||||||
- csi-vitastor-provisioner
|
|
||||||
topologyKey: "kubernetes.io/hostname"
|
|
||||||
serviceAccountName: vitastor-csi-provisioner
|
|
||||||
priorityClassName: system-cluster-critical
|
|
||||||
containers:
|
|
||||||
- name: csi-provisioner
|
|
||||||
image: k8s.gcr.io/sig-storage/csi-provisioner:v2.2.0
|
|
||||||
args:
|
|
||||||
- "--csi-address=$(ADDRESS)"
|
|
||||||
- "--v=5"
|
|
||||||
- "--timeout=150s"
|
|
||||||
- "--retry-interval-start=500ms"
|
|
||||||
- "--leader-election=true"
|
|
||||||
# set it to true to use topology based provisioning
|
|
||||||
- "--feature-gates=Topology=false"
|
|
||||||
# if fstype is not specified in storageclass, ext4 is default
|
|
||||||
- "--default-fstype=ext4"
|
|
||||||
- "--extra-create-metadata=true"
|
|
||||||
env:
|
|
||||||
- name: ADDRESS
|
|
||||||
value: unix:///csi/csi-provisioner.sock
|
|
||||||
imagePullPolicy: "IfNotPresent"
|
|
||||||
volumeMounts:
|
|
||||||
- name: socket-dir
|
|
||||||
mountPath: /csi
|
|
||||||
- name: csi-snapshotter
|
|
||||||
image: k8s.gcr.io/sig-storage/csi-snapshotter:v4.0.0
|
|
||||||
args:
|
|
||||||
- "--csi-address=$(ADDRESS)"
|
|
||||||
- "--v=5"
|
|
||||||
- "--timeout=150s"
|
|
||||||
- "--leader-election=true"
|
|
||||||
env:
|
|
||||||
- name: ADDRESS
|
|
||||||
value: unix:///csi/csi-provisioner.sock
|
|
||||||
imagePullPolicy: "IfNotPresent"
|
|
||||||
securityContext:
|
|
||||||
privileged: true
|
|
||||||
volumeMounts:
|
|
||||||
- name: socket-dir
|
|
||||||
mountPath: /csi
|
|
||||||
- name: csi-attacher
|
|
||||||
image: k8s.gcr.io/sig-storage/csi-attacher:v3.1.0
|
|
||||||
args:
|
|
||||||
- "--v=5"
|
|
||||||
- "--csi-address=$(ADDRESS)"
|
|
||||||
- "--leader-election=true"
|
|
||||||
- "--retry-interval-start=500ms"
|
|
||||||
env:
|
|
||||||
- name: ADDRESS
|
|
||||||
value: /csi/csi-provisioner.sock
|
|
||||||
imagePullPolicy: "IfNotPresent"
|
|
||||||
volumeMounts:
|
|
||||||
- name: socket-dir
|
|
||||||
mountPath: /csi
|
|
||||||
- name: csi-resizer
|
|
||||||
image: k8s.gcr.io/sig-storage/csi-resizer:v1.1.0
|
|
||||||
args:
|
|
||||||
- "--csi-address=$(ADDRESS)"
|
|
||||||
- "--v=5"
|
|
||||||
- "--timeout=150s"
|
|
||||||
- "--leader-election"
|
|
||||||
- "--retry-interval-start=500ms"
|
|
||||||
- "--handle-volume-inuse-error=false"
|
|
||||||
env:
|
|
||||||
- name: ADDRESS
|
|
||||||
value: unix:///csi/csi-provisioner.sock
|
|
||||||
imagePullPolicy: "IfNotPresent"
|
|
||||||
volumeMounts:
|
|
||||||
- name: socket-dir
|
|
||||||
mountPath: /csi
|
|
||||||
- name: csi-vitastor
|
|
||||||
securityContext:
|
|
||||||
privileged: true
|
|
||||||
capabilities:
|
|
||||||
add: ["SYS_ADMIN"]
|
|
||||||
image: vitalif/vitastor-csi:v0.6.10
|
|
||||||
args:
|
|
||||||
- "--node=$(NODE_ID)"
|
|
||||||
- "--endpoint=$(CSI_ENDPOINT)"
|
|
||||||
env:
|
|
||||||
- name: NODE_ID
|
|
||||||
valueFrom:
|
|
||||||
fieldRef:
|
|
||||||
fieldPath: spec.nodeName
|
|
||||||
- name: CSI_ENDPOINT
|
|
||||||
value: unix:///csi/csi-provisioner.sock
|
|
||||||
imagePullPolicy: "IfNotPresent"
|
|
||||||
volumeMounts:
|
|
||||||
- name: socket-dir
|
|
||||||
mountPath: /csi
|
|
||||||
- mountPath: /dev
|
|
||||||
name: host-dev
|
|
||||||
- mountPath: /sys
|
|
||||||
name: host-sys
|
|
||||||
- mountPath: /lib/modules
|
|
||||||
name: lib-modules
|
|
||||||
readOnly: true
|
|
||||||
- name: vitastor-config
|
|
||||||
mountPath: /etc/vitastor
|
|
||||||
volumes:
|
|
||||||
- name: host-dev
|
|
||||||
hostPath:
|
|
||||||
path: /dev
|
|
||||||
- name: host-sys
|
|
||||||
hostPath:
|
|
||||||
path: /sys
|
|
||||||
- name: lib-modules
|
|
||||||
hostPath:
|
|
||||||
path: /lib/modules
|
|
||||||
- name: socket-dir
|
|
||||||
emptyDir: {
|
|
||||||
medium: "Memory"
|
|
||||||
}
|
|
||||||
- name: vitastor-config
|
|
||||||
configMap:
|
|
||||||
name: vitastor-config
|
|
|
@ -1,11 +0,0 @@
|
||||||
---
|
|
||||||
# if Kubernetes version is less than 1.18 change
|
|
||||||
# apiVersion to storage.k8s.io/v1betav1
|
|
||||||
apiVersion: storage.k8s.io/v1
|
|
||||||
kind: CSIDriver
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: csi.vitastor.io
|
|
||||||
spec:
|
|
||||||
attachRequired: true
|
|
||||||
podInfoOnMount: false
|
|
|
@ -1,19 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: storage.k8s.io/v1
|
|
||||||
kind: StorageClass
|
|
||||||
metadata:
|
|
||||||
namespace: vitastor-system
|
|
||||||
name: vitastor
|
|
||||||
annotations:
|
|
||||||
storageclass.kubernetes.io/is-default-class: "true"
|
|
||||||
provisioner: csi.vitastor.io
|
|
||||||
volumeBindingMode: Immediate
|
|
||||||
parameters:
|
|
||||||
etcdVolumePrefix: ""
|
|
||||||
poolId: "1"
|
|
||||||
# you can choose other configuration file if you have it in the config map
|
|
||||||
#configPath: "/etc/vitastor/vitastor.conf"
|
|
||||||
# you can also specify etcdUrl here, maybe to connect to another Vitastor cluster
|
|
||||||
# multiple etcdUrls may be specified, delimited by comma
|
|
||||||
#etcdUrl: "http://192.168.7.2:2379"
|
|
||||||
#etcdPrefix: "/vitastor"
|
|
|
@ -1,12 +0,0 @@
|
||||||
---
|
|
||||||
apiVersion: v1
|
|
||||||
kind: PersistentVolumeClaim
|
|
||||||
metadata:
|
|
||||||
name: test-vitastor-pvc
|
|
||||||
spec:
|
|
||||||
storageClassName: vitastor
|
|
||||||
accessModes:
|
|
||||||
- ReadWriteOnce
|
|
||||||
resources:
|
|
||||||
requests:
|
|
||||||
storage: 10Gi
|
|
35
csi/go.mod
35
csi/go.mod
|
@ -1,35 +0,0 @@
|
||||||
module vitastor.io/csi
|
|
||||||
|
|
||||||
go 1.15
|
|
||||||
|
|
||||||
require (
|
|
||||||
github.com/container-storage-interface/spec v1.4.0
|
|
||||||
github.com/coreos/bbolt v0.0.0-00010101000000-000000000000 // indirect
|
|
||||||
github.com/coreos/etcd v3.3.25+incompatible // indirect
|
|
||||||
github.com/coreos/go-semver v0.3.0 // indirect
|
|
||||||
github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf // indirect
|
|
||||||
github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f // indirect
|
|
||||||
github.com/dustin/go-humanize v1.0.0 // indirect
|
|
||||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b
|
|
||||||
github.com/gorilla/websocket v1.4.2 // indirect
|
|
||||||
github.com/grpc-ecosystem/go-grpc-middleware v1.3.0 // indirect
|
|
||||||
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 // indirect
|
|
||||||
github.com/grpc-ecosystem/grpc-gateway v1.16.0 // indirect
|
|
||||||
github.com/jonboulle/clockwork v0.2.2 // indirect
|
|
||||||
github.com/kubernetes-csi/csi-lib-utils v0.9.1
|
|
||||||
github.com/soheilhy/cmux v0.1.5 // indirect
|
|
||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20201229170055-e5319fda7802 // indirect
|
|
||||||
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2 // indirect
|
|
||||||
go.etcd.io/bbolt v0.0.0-00010101000000-000000000000 // indirect
|
|
||||||
go.etcd.io/etcd v3.3.25+incompatible
|
|
||||||
golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb
|
|
||||||
google.golang.org/grpc v1.33.1
|
|
||||||
k8s.io/klog v1.0.0
|
|
||||||
k8s.io/utils v0.0.0-20210305010621-2afb4311ab10
|
|
||||||
)
|
|
||||||
|
|
||||||
replace github.com/coreos/bbolt => go.etcd.io/bbolt v1.3.5
|
|
||||||
|
|
||||||
replace go.etcd.io/bbolt => github.com/coreos/bbolt v1.3.5
|
|
||||||
|
|
||||||
replace google.golang.org/grpc => google.golang.org/grpc v1.25.1
|
|
448
csi/go.sum
448
csi/go.sum
|
@ -1,448 +0,0 @@
|
||||||
cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw=
|
|
||||||
cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU=
|
|
||||||
cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU=
|
|
||||||
cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY=
|
|
||||||
cloud.google.com/go v0.45.1/go.mod h1:RpBamKRgapWJb87xiFSdk4g1CME7QZg3uwTez+TSTjc=
|
|
||||||
cloud.google.com/go v0.46.3/go.mod h1:a6bKKbmY7er1mI7TEI4lsAkts/mkhTSZK8w33B4RAg0=
|
|
||||||
cloud.google.com/go v0.51.0/go.mod h1:hWtGJ6gnXH+KgDv+V0zFGDvpi07n3z8ZNj3T1RW0Gcw=
|
|
||||||
cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o=
|
|
||||||
cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE=
|
|
||||||
cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I=
|
|
||||||
cloud.google.com/go/storage v1.0.0/go.mod h1:IhtSnM/ZTZV8YYJWCY8RULGVqBDmpoyjwiyrjsg+URw=
|
|
||||||
dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU=
|
|
||||||
github.com/Azure/go-ansiterm v0.0.0-20170929234023-d6e3b3328b78/go.mod h1:LmzpDX56iTiv29bbRTIsUNlaFfuhWRQBWjQdVyAevI8=
|
|
||||||
github.com/Azure/go-autorest/autorest v0.9.0/go.mod h1:xyHB1BMZT0cuDHU7I0+g046+BFDTQ8rEZB0s4Yfa6bI=
|
|
||||||
github.com/Azure/go-autorest/autorest v0.9.6/go.mod h1:/FALq9T/kS7b5J5qsQ+RSTUdAmGFqi0vUdVNNx8q630=
|
|
||||||
github.com/Azure/go-autorest/autorest/adal v0.5.0/go.mod h1:8Z9fGy2MpX0PvDjB1pEgQTmVqjGhiHBW7RJJEciWzS0=
|
|
||||||
github.com/Azure/go-autorest/autorest/adal v0.8.2/go.mod h1:ZjhuQClTqx435SRJ2iMlOxPYt3d2C/T/7TiQCVZSn3Q=
|
|
||||||
github.com/Azure/go-autorest/autorest/date v0.1.0/go.mod h1:plvfp3oPSKwf2DNjlBjWF/7vwR+cUD/ELuzDCXwHUVA=
|
|
||||||
github.com/Azure/go-autorest/autorest/date v0.2.0/go.mod h1:vcORJHLJEh643/Ioh9+vPmf1Ij9AEBM5FuBIXLmIy0g=
|
|
||||||
github.com/Azure/go-autorest/autorest/mocks v0.1.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0=
|
|
||||||
github.com/Azure/go-autorest/autorest/mocks v0.2.0/go.mod h1:OTyCOPRA2IgIlWxVYxBee2F5Gr4kF2zd2J5cFRaIDN0=
|
|
||||||
github.com/Azure/go-autorest/autorest/mocks v0.3.0/go.mod h1:a8FDP3DYzQ4RYfVAxAN3SVSiiO77gL2j2ronKKP0syM=
|
|
||||||
github.com/Azure/go-autorest/logger v0.1.0/go.mod h1:oExouG+K6PryycPJfVSxi/koC6LSNgds39diKLz7Vrc=
|
|
||||||
github.com/Azure/go-autorest/tracing v0.5.0/go.mod h1:r/s2XiOKccPW3HrqB+W0TQzfbtp2fGCgRFtBroKn4Dk=
|
|
||||||
github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU=
|
|
||||||
github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo=
|
|
||||||
github.com/NYTimes/gziphandler v0.0.0-20170623195520-56545f4a5d46/go.mod h1:3wb06e3pkSAbeQ52E9H9iFoQsEEwGN64994WTCIhntQ=
|
|
||||||
github.com/PuerkitoBio/purell v1.0.0/go.mod h1:c11w/QuzBsJSee3cPx9rAFu61PvFxuPbtSwDGJws/X0=
|
|
||||||
github.com/PuerkitoBio/urlesc v0.0.0-20160726150825-5bd2802263f2/go.mod h1:uGdkoq3SwY9Y+13GIhn11/XLaGBb4BfwItxLd5jeuXE=
|
|
||||||
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
|
|
||||||
github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
|
|
||||||
github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
|
|
||||||
github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
|
|
||||||
github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY=
|
|
||||||
github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
|
|
||||||
github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8=
|
|
||||||
github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM=
|
|
||||||
github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw=
|
|
||||||
github.com/blang/semver v3.5.0+incompatible/go.mod h1:kRBLl5iJ+tD4TcOOxsy/0fnwebNt5EWlYSAyrTnjyyk=
|
|
||||||
github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU=
|
|
||||||
github.com/cespare/xxhash/v2 v2.1.1 h1:6MnRN8NT7+YBpUIWxHtefFZOKTAPgGjpQSxqLNn0+qY=
|
|
||||||
github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs=
|
|
||||||
github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI=
|
|
||||||
github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI=
|
|
||||||
github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU=
|
|
||||||
github.com/container-storage-interface/spec v1.2.0/go.mod h1:6URME8mwIBbpVyZV93Ce5St17xBiQJQY67NDsuohiy4=
|
|
||||||
github.com/container-storage-interface/spec v1.4.0 h1:ozAshSKxpJnYUfmkpZCTYyF/4MYeYlhdXbAvPvfGmkg=
|
|
||||||
github.com/container-storage-interface/spec v1.4.0/go.mod h1:6URME8mwIBbpVyZV93Ce5St17xBiQJQY67NDsuohiy4=
|
|
||||||
github.com/coreos/bbolt v1.3.5 h1:XFv7xaq7701j8ZSEzR28VohFYSlyakMyqNMU5FQH6Ac=
|
|
||||||
github.com/coreos/bbolt v1.3.5/go.mod h1:G5EMThwa9y8QZGBClrRx5EY+Yw9kAhnjy3bSjsnlVTQ=
|
|
||||||
github.com/coreos/etcd v3.3.25+incompatible h1:0GQEw6h3YnuOVdtwygkIfJ+Omx0tZ8/QkVyXI4LkbeY=
|
|
||||||
github.com/coreos/etcd v3.3.25+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE=
|
|
||||||
github.com/coreos/go-semver v0.3.0 h1:wkHLiw0WNATZnSG7epLsujiMCgPAc9xhjJ4tgnAxmfM=
|
|
||||||
github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk=
|
|
||||||
github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf h1:iW4rZ826su+pqaw19uhpSCzhj44qo35pNgKFGqzDKkU=
|
|
||||||
github.com/coreos/go-systemd v0.0.0-20191104093116-d3cd4ed1dbcf/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4=
|
|
||||||
github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f h1:lBNOc5arjvs8E5mO2tbpBpLoyyu8B6e44T7hJy6potg=
|
|
||||||
github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA=
|
|
||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
|
||||||
github.com/dgrijalva/jwt-go v3.2.0+incompatible h1:7qlOGliEKZXTDg6OTjfoBKDXWrumCAMpl/TFQ4/5kLM=
|
|
||||||
github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ=
|
|
||||||
github.com/docker/spdystream v0.0.0-20160310174837-449fdfce4d96/go.mod h1:Qh8CwZgvJUkLughtfhJv5dyTYa91l1fOUCrgjqmcifM=
|
|
||||||
github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815/go.mod h1:WwZ+bS3ebgob9U8Nd0kOddGdZWjyMGR8Wziv+TBNwSE=
|
|
||||||
github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo=
|
|
||||||
github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk=
|
|
||||||
github.com/elazarl/goproxy v0.0.0-20180725130230-947c36da3153/go.mod h1:/Zj4wYkgs4iZTTu3o/KG3Itv/qCCa8VVMlb3i9OVuzc=
|
|
||||||
github.com/emicklei/go-restful v0.0.0-20170410110728-ff4f55a20633/go.mod h1:otzb+WCGbkyDHkqmQmT5YD2WR4BBwUdeQoFo8l/7tVs=
|
|
||||||
github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4=
|
|
||||||
github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c=
|
|
||||||
github.com/evanphx/json-patch v4.9.0+incompatible/go.mod h1:50XU6AFN0ol/bzJsmQLiYLvXMP4fmwYFNcr97nuDLSk=
|
|
||||||
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
|
|
||||||
github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ=
|
|
||||||
github.com/ghodss/yaml v0.0.0-20150909031657-73d445a93680/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
|
|
||||||
github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04=
|
|
||||||
github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8=
|
|
||||||
github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
|
|
||||||
github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as=
|
|
||||||
github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE=
|
|
||||||
github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk=
|
|
||||||
github.com/go-logr/logr v0.1.0/go.mod h1:ixOQHD9gLJUVQQ2ZOR7zLEifBX6tGkNJF4QyIY7sIas=
|
|
||||||
github.com/go-logr/logr v0.2.0 h1:QvGt2nLcHH0WK9orKa+ppBPAxREcH364nPUedEpK0TY=
|
|
||||||
github.com/go-logr/logr v0.2.0/go.mod h1:z6/tIYblkpsD+a4lm/fGIIU9mZ+XfAiaFtq7xTgseGU=
|
|
||||||
github.com/go-openapi/jsonpointer v0.0.0-20160704185906-46af16f9f7b1/go.mod h1:+35s3my2LFTysnkMfxsJBAMHj/DoqoB9knIWoYG/Vk0=
|
|
||||||
github.com/go-openapi/jsonreference v0.0.0-20160704190145-13c6e3589ad9/go.mod h1:W3Z9FmVs9qj+KR4zFKmDPGiLdk1D9Rlm7cyMvf57TTg=
|
|
||||||
github.com/go-openapi/spec v0.0.0-20160808142527-6aced65f8501/go.mod h1:J8+jY1nAiCcj+friV/PDoE1/3eeccG9LYBs0tYvLOWc=
|
|
||||||
github.com/go-openapi/swag v0.0.0-20160704191624-1d0bd113de87/go.mod h1:DXUve3Dpr1UfpPtxFw+EFuQ41HhCWZfha5jSVRG7C7I=
|
|
||||||
github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY=
|
|
||||||
github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ=
|
|
||||||
github.com/gogo/protobuf v1.3.1 h1:DqDEcV5aeaTmdFBePNpYsp3FlcVH/2ISVVM9Qf8PSls=
|
|
||||||
github.com/gogo/protobuf v1.3.1/go.mod h1:SlYgWuQ5SjCEi6WLHjHCa1yvBfUnHcTbrrZtXPKa29o=
|
|
||||||
github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q=
|
|
||||||
github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q=
|
|
||||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b h1:VKtxabqXZkF25pY9ekfRL6a582T4P37/31XEstQ5p58=
|
|
||||||
github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q=
|
|
||||||
github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
|
||||||
github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7 h1:5ZkaAPbicIKTF2I64qf5Fh8Aa83Q/dnOafMYV0OMwjA=
|
|
||||||
github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc=
|
|
||||||
github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
|
||||||
github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A=
|
|
||||||
github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y=
|
|
||||||
github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
|
||||||
github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
|
||||||
github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
|
|
||||||
github.com/golang/protobuf v1.3.3/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw=
|
|
||||||
github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8=
|
|
||||||
github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA=
|
|
||||||
github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs=
|
|
||||||
github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w=
|
|
||||||
github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0=
|
|
||||||
github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8=
|
|
||||||
github.com/golang/protobuf v1.4.2 h1:+Z5KGCizgyZCbGh1KZqA0fcLLkwbsjIzS4aV2v7wJX0=
|
|
||||||
github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI=
|
|
||||||
github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
|
||||||
github.com/google/btree v1.0.0 h1:0udJVsspx3VBr5FwtLhQQtuAsVc79tTq0ocGIPAU6qo=
|
|
||||||
github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ=
|
|
||||||
github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M=
|
|
||||||
github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
|
||||||
github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU=
|
|
||||||
github.com/google/go-cmp v0.4.0 h1:xsAVV57WRhGj6kEIi8ReJzQlHHqcBYCElAvkovg3B/4=
|
|
||||||
github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
|
|
||||||
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
|
|
||||||
github.com/google/gofuzz v1.1.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
|
|
||||||
github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs=
|
|
||||||
github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
|
||||||
github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc=
|
|
||||||
github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM=
|
|
||||||
github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI=
|
|
||||||
github.com/google/uuid v1.1.1 h1:Gkbcsh/GbpXz7lPftLA3P6TYMwjCLYm83jiFQZF/3gY=
|
|
||||||
github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
|
||||||
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
|
||||||
github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
|
|
||||||
github.com/googleapis/gnostic v0.4.1/go.mod h1:LRhVm6pbyptWbWbuZ38d1eyptfvIytN3ir6b65WBswg=
|
|
||||||
github.com/gorilla/websocket v1.4.2 h1:+/TMaTYc4QFitKJxsQ7Yye35DkWvkdLcvGKqM+x0Ufc=
|
|
||||||
github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
|
|
||||||
github.com/gregjones/httpcache v0.0.0-20180305231024-9cad4c3443a7/go.mod h1:FecbI9+v66THATjSRHfNgh1IVFe/9kFxbXtjV0ctIMA=
|
|
||||||
github.com/grpc-ecosystem/go-grpc-middleware v1.3.0 h1:+9834+KizmvFV7pXQGSXQTsaWhq2GjuNUt0aUU0YBYw=
|
|
||||||
github.com/grpc-ecosystem/go-grpc-middleware v1.3.0/go.mod h1:z0ButlSOZa5vEBq9m2m2hlwIgKw+rp3sdCBRoJY+30Y=
|
|
||||||
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0 h1:Ovs26xHkKqVztRpIrF/92BcuyuQ/YW4NSIpoGtfXNho=
|
|
||||||
github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk=
|
|
||||||
github.com/grpc-ecosystem/grpc-gateway v1.16.0 h1:gmcG1KaJ57LophUzW0Hy8NmPhnMZb4M0+kPpLofRdBo=
|
|
||||||
github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw=
|
|
||||||
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
|
||||||
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
|
||||||
github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU=
|
|
||||||
github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc=
|
|
||||||
github.com/imdario/mergo v0.3.5/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA=
|
|
||||||
github.com/jonboulle/clockwork v0.2.2 h1:UOGuzwb1PwsrDAObMuhUnj0p5ULPj8V/xJ7Kx9qUBdQ=
|
|
||||||
github.com/jonboulle/clockwork v0.2.2/go.mod h1:Pkfl5aHPm1nk2H9h0bjmnJD/BcgbGXUBGnn1kMkgxc8=
|
|
||||||
github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
|
|
||||||
github.com/json-iterator/go v1.1.10 h1:Kz6Cvnvv2wGdaG/V8yMvfkmNiXq9Ya2KUv4rouJJr68=
|
|
||||||
github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
|
|
||||||
github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU=
|
|
||||||
github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk=
|
|
||||||
github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w=
|
|
||||||
github.com/kisielk/errcheck v1.2.0/go.mod h1:/BMXB+zMLi60iA8Vv6Ksmxu/1UDYcXs4uQLJ+jE2L00=
|
|
||||||
github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8=
|
|
||||||
github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck=
|
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.3 h1:CE8S1cTafDpPvMhIxNJKvHsGVBgn1xWYf1NbHQhywc8=
|
|
||||||
github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ=
|
|
||||||
github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc=
|
|
||||||
github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo=
|
|
||||||
github.com/kr/pretty v0.2.0 h1:s5hAObm+yFO5uHYt5dYjxi2rXrsnmRpJx4OYvIWUaQs=
|
|
||||||
github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI=
|
|
||||||
github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ=
|
|
||||||
github.com/kr/text v0.1.0 h1:45sCR5RtlFHMR4UwH9sdQ5TC8v0qDQCHnXt+kaKSTVE=
|
|
||||||
github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI=
|
|
||||||
github.com/kubernetes-csi/csi-lib-utils v0.9.1 h1:sGq6ifVujfMSkfTsMZip44Ttv8SDXvsBlFk9GdYl/b8=
|
|
||||||
github.com/kubernetes-csi/csi-lib-utils v0.9.1/go.mod h1:8E2jVUX9j3QgspwHXa6LwyN7IHQDjW9jX3kwoWnSC+M=
|
|
||||||
github.com/mailru/easyjson v0.0.0-20160728113105-d5b7844b561a/go.mod h1:C1wdFJiN94OJF2b5HbByQZoLdCWB1Yqtg26g4irojpc=
|
|
||||||
github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0=
|
|
||||||
github.com/matttproud/golang_protobuf_extensions v1.0.2-0.20181231171920-c182affec369 h1:I0XW9+e1XWDxdcEniV4rQAIOPUGDq67JSCiRCgGCZLI=
|
|
||||||
github.com/matttproud/golang_protobuf_extensions v1.0.2-0.20181231171920-c182affec369/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4=
|
|
||||||
github.com/moby/term v0.0.0-20200312100748-672ec06f55cd/go.mod h1:DdlQx2hp0Ss5/fLikoLlEeIYiATotOjgB//nb973jeo=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
|
|
||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
|
|
||||||
github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
|
||||||
github.com/modern-go/reflect2 v1.0.1 h1:9f412s+6RmYXLWZSEzVVgPGK7C2PphHj5RJrvfx9AWI=
|
|
||||||
github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
|
|
||||||
github.com/munnerz/goautoneg v0.0.0-20120707110453-a547fc61f48d/go.mod h1:+n7T8mK8HuQTcFwEeznm/DIxMOiR9yIdICNftLE1DvQ=
|
|
||||||
github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U=
|
|
||||||
github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f/go.mod h1:ZdcZmHo+o7JKHSa8/e818NopupXU1YMK5fe1lsApnBw=
|
|
||||||
github.com/onsi/ginkgo v0.0.0-20170829012221-11459a886d9c/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
|
||||||
github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
|
||||||
github.com/onsi/ginkgo v1.11.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE=
|
|
||||||
github.com/onsi/gomega v0.0.0-20170829124025-dcabb60a477c/go.mod h1:C1qb7wdrVGGVU+Z6iS04AVkA3Q65CEZX59MT0QO5uiA=
|
|
||||||
github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY=
|
|
||||||
github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o=
|
|
||||||
github.com/peterbourgon/diskv v2.0.1+incompatible/go.mod h1:uqqh8zWWbv1HBMNONnaR/tNboyR3/BZd58JJSHlUSCU=
|
|
||||||
github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
|
||||||
github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
|
||||||
github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4=
|
|
||||||
github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
|
|
||||||
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
|
|
||||||
github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw=
|
|
||||||
github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo=
|
|
||||||
github.com/prometheus/client_golang v1.7.1 h1:NTGy1Ja9pByO+xAeH/qiWnLrKtr3hJPNjaVUwnjpdpA=
|
|
||||||
github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M=
|
|
||||||
github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo=
|
|
||||||
github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
|
||||||
github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
|
||||||
github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M=
|
|
||||||
github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA=
|
|
||||||
github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4=
|
|
||||||
github.com/prometheus/common v0.10.0 h1:RyRA7RzGXQZiW+tGMr7sxa85G1z0yOpM1qq5c8lNawc=
|
|
||||||
github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo=
|
|
||||||
github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk=
|
|
||||||
github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA=
|
|
||||||
github.com/prometheus/procfs v0.1.3 h1:F0+tqvhOksq22sc6iCHF5WGlWjdwj92p0udFh1VFBS8=
|
|
||||||
github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU=
|
|
||||||
github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ=
|
|
||||||
github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4=
|
|
||||||
github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo=
|
|
||||||
github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE=
|
|
||||||
github.com/sirupsen/logrus v1.6.0 h1:UBcNElsrwanuuMsnGSlYmtmgbb23qDR5dG+6X6Oo89I=
|
|
||||||
github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88=
|
|
||||||
github.com/soheilhy/cmux v0.1.5 h1:jjzc5WVemNEDTLwv9tlmemhC73tI08BNOIGwBOo10Js=
|
|
||||||
github.com/soheilhy/cmux v0.1.5/go.mod h1:T7TcVDs9LWfQgPlPsdngu6I6QIoyIFZDDC6sNE1GqG0=
|
|
||||||
github.com/spf13/afero v1.2.2/go.mod h1:9ZxEEn6pIJ8Rxe320qSDBk6AsU0r9pR7Q4OcevTdifk=
|
|
||||||
github.com/spf13/pflag v0.0.0-20170130214245-9ff6c6923cff/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
|
|
||||||
github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4=
|
|
||||||
github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg=
|
|
||||||
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
|
||||||
github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
|
|
||||||
github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs=
|
|
||||||
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
|
|
||||||
github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
|
|
||||||
github.com/stretchr/testify v1.5.1 h1:nOGnQDM7FYENwehXlg/kFVnos3rEvtKTjRvOWSzb6H4=
|
|
||||||
github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA=
|
|
||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20201229170055-e5319fda7802 h1:uruHq4dN7GR16kFc5fp3d1RIYzJW5onx8Ybykw2YQFA=
|
|
||||||
github.com/tmc/grpc-websocket-proxy v0.0.0-20201229170055-e5319fda7802/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U=
|
|
||||||
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2 h1:eY9dn8+vbi4tKz5Qo6v2eYzo7kUS51QINcR5jNpbZS8=
|
|
||||||
github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU=
|
|
||||||
github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
|
||||||
github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74=
|
|
||||||
go.etcd.io/bbolt v1.3.5 h1:XAzx9gjCb0Rxj7EoqcClPD1d5ZBxZJk0jbuoPHenBt0=
|
|
||||||
go.etcd.io/bbolt v1.3.5/go.mod h1:G5EMThwa9y8QZGBClrRx5EY+Yw9kAhnjy3bSjsnlVTQ=
|
|
||||||
go.etcd.io/etcd v3.3.25+incompatible h1:V1RzkZJj9LqsJRy+TUBgpWSbZXITLB819lstuTFoZOY=
|
|
||||||
go.etcd.io/etcd v3.3.25+incompatible/go.mod h1:yaeTdrJi5lOmYerz05bd8+V7KubZs8YSFZfzsF9A6aI=
|
|
||||||
go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU=
|
|
||||||
go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8=
|
|
||||||
go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw=
|
|
||||||
go.uber.org/atomic v1.4.0 h1:cxzIVoETapQEqDhQu3QfnvXAV4AlzcvUCxkVUFw3+EU=
|
|
||||||
go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE=
|
|
||||||
go.uber.org/multierr v1.1.0 h1:HoEmRHQPVSqub6w2z2d2EOVs2fjyFRGyofhKuyDq0QI=
|
|
||||||
go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0=
|
|
||||||
go.uber.org/zap v1.10.0 h1:ORx85nbTijNz8ljznvCMR1ZBIPKFn3jQrag10X2AsuM=
|
|
||||||
go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q=
|
|
||||||
golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4=
|
|
||||||
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
|
|
||||||
golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
|
||||||
golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
|
||||||
golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI=
|
|
||||||
golang.org/x/crypto v0.0.0-20191206172530-e9b2fee46413/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
|
||||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9 h1:psW17arqaxU48Z5kZ0CQnkZWQJsqcURM6tKiBApRjXI=
|
|
||||||
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
|
|
||||||
golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
|
||||||
golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA=
|
|
||||||
golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8=
|
|
||||||
golang.org/x/exp v0.0.0-20190829153037-c13cbed26979/go.mod h1:86+5VVa7VpoJ4kLfm080zCjGlMRFzhUhsZKEZO7MGek=
|
|
||||||
golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4=
|
|
||||||
golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js=
|
|
||||||
golang.org/x/image v0.0.0-20190802002840-cff245a6509b/go.mod h1:FeLwcggjj3mMvU+oOTbSwawSJRM1uh48EjtB4UJZlP0=
|
|
||||||
golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU=
|
|
||||||
golang.org/x/lint v0.0.0-20190301231843-5614ed5bae6f/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE=
|
|
||||||
golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
|
||||||
golang.org/x/lint v0.0.0-20190409202823-959b441ac422/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
|
||||||
golang.org/x/lint v0.0.0-20190909230951-414d861bb4ac/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc=
|
|
||||||
golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs=
|
|
||||||
golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE=
|
|
||||||
golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o=
|
|
||||||
golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc=
|
|
||||||
golang.org/x/mod v0.1.0/go.mod h1:0QHyrYULN0/3qlju5TqG8bIK38QM8yzMo5ekMj3DlcY=
|
|
||||||
golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg=
|
|
||||||
golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
|
||||||
golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA=
|
|
||||||
golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4=
|
|
||||||
golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg=
|
|
||||||
golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks=
|
|
||||||
golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
|
||||||
golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
|
||||||
golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
|
||||||
golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s=
|
|
||||||
golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A=
|
|
||||||
golang.org/x/net v0.0.0-20200707034311-ab3426394381 h1:VXak5I6aEWmAXeQjA+QSZzlgNrpq9mjcfDemuexIKsU=
|
|
||||||
golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
|
|
||||||
golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA=
|
|
||||||
golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
|
|
||||||
golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb h1:eBmm0M9fYhWpKZLjQUUKka/LtIxf46G4fxeEz5KJr9U=
|
|
||||||
golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU=
|
|
||||||
golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U=
|
|
||||||
golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
|
||||||
golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
|
||||||
golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
|
||||||
golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw=
|
|
||||||
golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sync v0.0.0-20190227155943-e225da77a7e6/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
|
|
||||||
golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
|
|
||||||
golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200622214017-ed371f2e16b4 h1:5/PjkGUjvEU5Gl6BxmvKRPpqo2uNMv4rcHBMwzk/st8=
|
|
||||||
golang.org/x/sys v0.0.0-20200622214017-ed371f2e16b4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f h1:+Nyd8tzPX9R7BWHguqsrbFdRx3WQ/1ib8I44HXV5yTA=
|
|
||||||
golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
|
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
|
||||||
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
|
||||||
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
|
||||||
golang.org/x/text v0.3.3 h1:cokOdA+Jmi5PJGXLlLllQSgYigAEfHXJAERHVMaCc2k=
|
|
||||||
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
|
|
||||||
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
|
||||||
golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
|
||||||
golang.org/x/time v0.0.0-20191024005414-555d28b269f0 h1:/5xXl8Y5W96D+TtHSlonuFqGHIWVuyCkGJLwGh9JJFs=
|
|
||||||
golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
|
||||||
golang.org/x/tools v0.0.0-20181011042414-1f849cf54d09/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
|
||||||
golang.org/x/tools v0.0.0-20181030221726-6c7e314b6563/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
|
||||||
golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY=
|
|
||||||
golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
|
||||||
golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
|
||||||
golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs=
|
|
||||||
golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
|
||||||
golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
|
||||||
golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q=
|
|
||||||
golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
|
||||||
golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
|
||||||
golang.org/x/tools v0.0.0-20190624222133-a101b041ded4/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
|
||||||
golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc=
|
|
||||||
golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo=
|
|
||||||
golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28=
|
|
||||||
golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE=
|
|
||||||
golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA=
|
|
||||||
golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543 h1:E7g+9GITq07hpfrRu66IVDexMakfv52eLZ2CXBWiKr4=
|
|
||||||
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 h1:go1bK/D/BFZV2I8cIQd1NKEZ+0owSTG1fDTci4IqFcE=
|
|
||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
|
|
||||||
google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE=
|
|
||||||
google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M=
|
|
||||||
google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
|
|
||||||
google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg=
|
|
||||||
google.golang.org/api v0.15.0/go.mod h1:iLdEw5Ide6rF15KTC1Kkl0iskquN2gFfn9o9XIsbkAI=
|
|
||||||
google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
|
||||||
google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4=
|
|
||||||
google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0=
|
|
||||||
google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc=
|
|
||||||
google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
|
||||||
google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
|
||||||
google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
|
||||||
google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE=
|
|
||||||
google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
|
|
||||||
google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc=
|
|
||||||
google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8=
|
|
||||||
google.golang.org/genproto v0.0.0-20191230161307-f3c370f40bfb/go.mod h1:n3cpQtvxv34hfy77yVDNjmbRyujviMdxYliBSkLhpCc=
|
|
||||||
google.golang.org/genproto v0.0.0-20200423170343-7949de9c1215/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
|
||||||
google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c=
|
|
||||||
google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013 h1:+kGHl1aib/qcwaRi1CbqBZ1rk19r85MNUf8HaBghugY=
|
|
||||||
google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo=
|
|
||||||
google.golang.org/grpc v1.25.1 h1:wdKvqQk7IttEw92GoRyKG2IDrUIpgpj6H6m81yfeMW0=
|
|
||||||
google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY=
|
|
||||||
google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8=
|
|
||||||
google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0=
|
|
||||||
google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM=
|
|
||||||
google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE=
|
|
||||||
google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo=
|
|
||||||
google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
|
||||||
google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
|
||||||
google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU=
|
|
||||||
google.golang.org/protobuf v1.24.0 h1:UhZDfRO8JRQru4/+LlLE0BRKGF8L+PICnvYZmx/fEGA=
|
|
||||||
google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4=
|
|
||||||
gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw=
|
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
|
||||||
gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
|
||||||
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15 h1:YR8cESwS4TdDjEe65xsg0ogRM/Nc3DYOhEAlW+xobZo=
|
|
||||||
gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
|
||||||
gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI=
|
|
||||||
gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys=
|
|
||||||
gopkg.in/inf.v0 v0.9.1/go.mod h1:cWUDdTG/fYaXco+Dcufb5Vnc6Gp2YChqWtbxRZE0mXw=
|
|
||||||
gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw=
|
|
||||||
gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8 h1:obN1ZagJSUGI0Ek/LBmuj4SNLPfIny3KsKFopxRdj10=
|
|
||||||
gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
|
|
||||||
gotest.tools v2.2.0+incompatible/go.mod h1:DsYFclhRJ6vuDpmuTbkuFWG+y2sxOXAzmJt81HFBacw=
|
|
||||||
gotest.tools/v3 v3.0.2/go.mod h1:3SzNCllyD9/Y+b5r9JIKQ474KzkZyqLqEfYqMsX94Bk=
|
|
||||||
honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
|
||||||
honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
|
||||||
honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
|
||||||
honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4=
|
|
||||||
honnef.co/go/tools v0.0.1-2019.2.3/go.mod h1:a3bituU0lyd329TUQxRnasdCoJDkEUEAqEt0JzvZhAg=
|
|
||||||
k8s.io/api v0.19.0/go.mod h1:I1K45XlvTrDjmj5LoM5LuP/KYrhWbjUKT/SoPG0qTjw=
|
|
||||||
k8s.io/apimachinery v0.19.0/go.mod h1:DnPGDnARWFvYa3pMHgSxtbZb7gpzzAZ1pTfaUNDVlmA=
|
|
||||||
k8s.io/client-go v0.19.0/go.mod h1:H9E/VT95blcFQnlyShFgnFT9ZnJOAceiUHM3MlRC+mU=
|
|
||||||
k8s.io/component-base v0.19.0/go.mod h1:dKsY8BxkA+9dZIAh2aWJLL/UdASFDNtGYTCItL4LM7Y=
|
|
||||||
k8s.io/gengo v0.0.0-20200413195148-3a45101e95ac/go.mod h1:ezvh/TsK7cY6rbqRK0oQQ8IAqLxYwwyPxAX1Pzy0ii0=
|
|
||||||
k8s.io/klog v1.0.0 h1:Pt+yjF5aB1xDSVbau4VsWe+dQNzA0qv1LlXdC2dF6Q8=
|
|
||||||
k8s.io/klog v1.0.0/go.mod h1:4Bi6QPql/J/LkTDqv7R/cd3hPo4k2DG6Ptcz060Ez5I=
|
|
||||||
k8s.io/klog/v2 v2.0.0/go.mod h1:PBfzABfn139FHAV07az/IF9Wp1bkk3vpT2XSJ76fSDE=
|
|
||||||
k8s.io/klog/v2 v2.2.0 h1:XRvcwJozkgZ1UQJmfMGpvRthQHOvihEhYtDfAaxMz/A=
|
|
||||||
k8s.io/klog/v2 v2.2.0/go.mod h1:Od+F08eJP+W3HUb4pSrPpgp9DGU4GzlpG/TmITuYh/Y=
|
|
||||||
k8s.io/kube-openapi v0.0.0-20200805222855-6aeccd4b50c6/go.mod h1:UuqjUnNftUyPE5H64/qeyjQoUZhGpeFDVdxjTeEVN2o=
|
|
||||||
k8s.io/utils v0.0.0-20200729134348-d5654de09c73/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA=
|
|
||||||
k8s.io/utils v0.0.0-20210305010621-2afb4311ab10 h1:u5rPykqiCpL+LBfjRkXvnK71gOgIdmq3eHUEkPrbeTI=
|
|
||||||
k8s.io/utils v0.0.0-20210305010621-2afb4311ab10/go.mod h1:jPW/WVKK9YHAvNhRxK0md/EJ228hCsBRufyofKtW8HA=
|
|
||||||
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
|
|
||||||
sigs.k8s.io/structured-merge-diff/v4 v4.0.1/go.mod h1:bJZC9H9iH24zzfZ/41RGcq60oK1F7G282QMXDPYydCw=
|
|
||||||
sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o=
|
|
||||||
sigs.k8s.io/yaml v1.2.0 h1:kr/MCeFWJWTwyaHoR9c8EjH9OumOmoF9YGiZd7lFm/Q=
|
|
||||||
sigs.k8s.io/yaml v1.2.0/go.mod h1:yfXDCHCao9+ENCvLSE62v9VSji2MKu5jeNfTrofGhJc=
|
|
|
@ -1,22 +0,0 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
|
||||||
// License: VNPL-1.1 or GNU GPL-2.0+ (see README.md for details)
|
|
||||||
|
|
||||||
package vitastor
|
|
||||||
|
|
||||||
const (
|
|
||||||
vitastorCSIDriverName = "csi.vitastor.io"
|
|
||||||
vitastorCSIDriverVersion = "0.6.10"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Config struct fills the parameters of request or user input
|
|
||||||
type Config struct
|
|
||||||
{
|
|
||||||
Endpoint string
|
|
||||||
NodeID string
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewConfig returns config struct to initialize new driver
|
|
||||||
func NewConfig() *Config
|
|
||||||
{
|
|
||||||
return &Config{}
|
|
||||||
}
|
|
|
@ -1,530 +0,0 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
|
||||||
// License: VNPL-1.1 or GNU GPL-2.0+ (see README.md for details)
|
|
||||||
|
|
||||||
package vitastor
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"encoding/json"
|
|
||||||
"strings"
|
|
||||||
"bytes"
|
|
||||||
"strconv"
|
|
||||||
"time"
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
"io/ioutil"
|
|
||||||
|
|
||||||
"github.com/kubernetes-csi/csi-lib-utils/protosanitizer"
|
|
||||||
"k8s.io/klog"
|
|
||||||
|
|
||||||
"google.golang.org/grpc/codes"
|
|
||||||
"google.golang.org/grpc/status"
|
|
||||||
|
|
||||||
"go.etcd.io/etcd/clientv3"
|
|
||||||
|
|
||||||
"github.com/container-storage-interface/spec/lib/go/csi"
|
|
||||||
)
|
|
||||||
|
|
||||||
const (
|
|
||||||
KB int64 = 1024
|
|
||||||
MB int64 = 1024 * KB
|
|
||||||
GB int64 = 1024 * MB
|
|
||||||
TB int64 = 1024 * GB
|
|
||||||
ETCD_TIMEOUT time.Duration = 15*time.Second
|
|
||||||
)
|
|
||||||
|
|
||||||
type InodeIndex struct
|
|
||||||
{
|
|
||||||
Id uint64 `json:"id"`
|
|
||||||
PoolId uint64 `json:"pool_id"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type InodeConfig struct
|
|
||||||
{
|
|
||||||
Name string `json:"name"`
|
|
||||||
Size uint64 `json:"size,omitempty"`
|
|
||||||
ParentPool uint64 `json:"parent_pool,omitempty"`
|
|
||||||
ParentId uint64 `json:"parent_id,omitempty"`
|
|
||||||
Readonly bool `json:"readonly,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type ControllerServer struct
|
|
||||||
{
|
|
||||||
*Driver
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewControllerServer create new instance controller
|
|
||||||
func NewControllerServer(driver *Driver) *ControllerServer
|
|
||||||
{
|
|
||||||
return &ControllerServer{
|
|
||||||
Driver: driver,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func GetConnectionParams(params map[string]string) (map[string]string, []string, string)
|
|
||||||
{
|
|
||||||
ctxVars := make(map[string]string)
|
|
||||||
configPath := params["configPath"]
|
|
||||||
if (configPath == "")
|
|
||||||
{
|
|
||||||
configPath = "/etc/vitastor/vitastor.conf"
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
ctxVars["configPath"] = configPath
|
|
||||||
}
|
|
||||||
config := make(map[string]interface{})
|
|
||||||
if configFD, err := os.Open(configPath); err == nil
|
|
||||||
{
|
|
||||||
defer configFD.Close()
|
|
||||||
data, _ := ioutil.ReadAll(configFD)
|
|
||||||
json.Unmarshal(data, &config)
|
|
||||||
}
|
|
||||||
// Try to load prefix & etcd URL from the config
|
|
||||||
var etcdUrl []string
|
|
||||||
if (params["etcdUrl"] != "")
|
|
||||||
{
|
|
||||||
ctxVars["etcdUrl"] = params["etcdUrl"]
|
|
||||||
etcdUrl = strings.Split(params["etcdUrl"], ",")
|
|
||||||
}
|
|
||||||
if (len(etcdUrl) == 0)
|
|
||||||
{
|
|
||||||
switch config["etcd_address"].(type)
|
|
||||||
{
|
|
||||||
case string:
|
|
||||||
etcdUrl = strings.Split(config["etcd_address"].(string), ",")
|
|
||||||
case []string:
|
|
||||||
etcdUrl = config["etcd_address"].([]string)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
etcdPrefix := params["etcdPrefix"]
|
|
||||||
if (etcdPrefix == "")
|
|
||||||
{
|
|
||||||
etcdPrefix, _ = config["etcd_prefix"].(string)
|
|
||||||
if (etcdPrefix == "")
|
|
||||||
{
|
|
||||||
etcdPrefix = "/vitastor"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
ctxVars["etcdPrefix"] = etcdPrefix
|
|
||||||
}
|
|
||||||
return ctxVars, etcdUrl, etcdPrefix
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create the volume
|
|
||||||
func (cs *ControllerServer) CreateVolume(ctx context.Context, req *csi.CreateVolumeRequest) (*csi.CreateVolumeResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received controller create volume request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
if (req == nil)
|
|
||||||
{
|
|
||||||
return nil, status.Errorf(codes.InvalidArgument, "request cannot be empty")
|
|
||||||
}
|
|
||||||
if (req.GetName() == "")
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "name is a required field")
|
|
||||||
}
|
|
||||||
volumeCapabilities := req.GetVolumeCapabilities()
|
|
||||||
if (volumeCapabilities == nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "volume capabilities is a required field")
|
|
||||||
}
|
|
||||||
|
|
||||||
etcdVolumePrefix := req.Parameters["etcdVolumePrefix"]
|
|
||||||
poolId, _ := strconv.ParseUint(req.Parameters["poolId"], 10, 64)
|
|
||||||
if (poolId == 0)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "poolId is missing in storage class configuration")
|
|
||||||
}
|
|
||||||
|
|
||||||
volName := etcdVolumePrefix + req.GetName()
|
|
||||||
volSize := 1 * GB
|
|
||||||
if capRange := req.GetCapacityRange(); capRange != nil
|
|
||||||
{
|
|
||||||
volSize = ((capRange.GetRequiredBytes() + MB - 1) / MB) * MB
|
|
||||||
}
|
|
||||||
|
|
||||||
// FIXME: The following should PROBABLY be implemented externally in a management tool
|
|
||||||
|
|
||||||
ctxVars, etcdUrl, etcdPrefix := GetConnectionParams(req.Parameters)
|
|
||||||
if (len(etcdUrl) == 0)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "no etcdUrl in storage class configuration and no etcd_address in vitastor.conf")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Connect to etcd
|
|
||||||
cli, err := clientv3.New(clientv3.Config{
|
|
||||||
DialTimeout: ETCD_TIMEOUT,
|
|
||||||
Endpoints: etcdUrl,
|
|
||||||
})
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to connect to etcd at "+strings.Join(etcdUrl, ",")+": "+err.Error())
|
|
||||||
}
|
|
||||||
defer cli.Close()
|
|
||||||
|
|
||||||
var imageId uint64 = 0
|
|
||||||
for
|
|
||||||
{
|
|
||||||
// Check if the image exists
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), ETCD_TIMEOUT)
|
|
||||||
resp, err := cli.Get(ctx, etcdPrefix+"/index/image/"+volName)
|
|
||||||
cancel()
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to read key from etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
if (len(resp.Kvs) > 0)
|
|
||||||
{
|
|
||||||
kv := resp.Kvs[0]
|
|
||||||
var v InodeIndex
|
|
||||||
err := json.Unmarshal(kv.Value, &v)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "invalid /index/image/"+volName+" key in etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
poolId = v.PoolId
|
|
||||||
imageId = v.Id
|
|
||||||
inodeCfgKey := fmt.Sprintf("/config/inode/%d/%d", poolId, imageId)
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), ETCD_TIMEOUT)
|
|
||||||
resp, err := cli.Get(ctx, etcdPrefix+inodeCfgKey)
|
|
||||||
cancel()
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to read key from etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
if (len(resp.Kvs) == 0)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "missing "+inodeCfgKey+" key in etcd")
|
|
||||||
}
|
|
||||||
var inodeCfg InodeConfig
|
|
||||||
err = json.Unmarshal(resp.Kvs[0].Value, &inodeCfg)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "invalid "+inodeCfgKey+" key in etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
if (inodeCfg.Size < uint64(volSize))
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "image "+volName+" is already created, but size is less than expected")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Find a free ID
|
|
||||||
// Create image metadata in a transaction verifying that the image doesn't exist yet AND ID is still free
|
|
||||||
maxIdKey := fmt.Sprintf("%s/index/maxid/%d", etcdPrefix, poolId)
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), ETCD_TIMEOUT)
|
|
||||||
resp, err := cli.Get(ctx, maxIdKey)
|
|
||||||
cancel()
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to read key from etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
var modRev int64
|
|
||||||
var nextId uint64
|
|
||||||
if (len(resp.Kvs) > 0)
|
|
||||||
{
|
|
||||||
var err error
|
|
||||||
nextId, err = strconv.ParseUint(string(resp.Kvs[0].Value), 10, 64)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, maxIdKey+" contains invalid ID")
|
|
||||||
}
|
|
||||||
modRev = resp.Kvs[0].ModRevision
|
|
||||||
nextId++
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
nextId = 1
|
|
||||||
}
|
|
||||||
inodeIdxJson, _ := json.Marshal(InodeIndex{
|
|
||||||
Id: nextId,
|
|
||||||
PoolId: poolId,
|
|
||||||
})
|
|
||||||
inodeCfgJson, _ := json.Marshal(InodeConfig{
|
|
||||||
Name: volName,
|
|
||||||
Size: uint64(volSize),
|
|
||||||
})
|
|
||||||
ctx, cancel = context.WithTimeout(context.Background(), ETCD_TIMEOUT)
|
|
||||||
txnResp, err := cli.Txn(ctx).If(
|
|
||||||
clientv3.Compare(clientv3.ModRevision(fmt.Sprintf("%s/index/maxid/%d", etcdPrefix, poolId)), "=", modRev),
|
|
||||||
clientv3.Compare(clientv3.CreateRevision(fmt.Sprintf("%s/index/image/%s", etcdPrefix, volName)), "=", 0),
|
|
||||||
clientv3.Compare(clientv3.CreateRevision(fmt.Sprintf("%s/config/inode/%d/%d", etcdPrefix, poolId, nextId)), "=", 0),
|
|
||||||
).Then(
|
|
||||||
clientv3.OpPut(fmt.Sprintf("%s/index/maxid/%d", etcdPrefix, poolId), fmt.Sprintf("%d", nextId)),
|
|
||||||
clientv3.OpPut(fmt.Sprintf("%s/index/image/%s", etcdPrefix, volName), string(inodeIdxJson)),
|
|
||||||
clientv3.OpPut(fmt.Sprintf("%s/config/inode/%d/%d", etcdPrefix, poolId, nextId), string(inodeCfgJson)),
|
|
||||||
).Commit()
|
|
||||||
cancel()
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to commit transaction in etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
if (txnResp.Succeeded)
|
|
||||||
{
|
|
||||||
imageId = nextId
|
|
||||||
break
|
|
||||||
}
|
|
||||||
// Start over if the transaction fails
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ctxVars["name"] = volName
|
|
||||||
volumeIdJson, _ := json.Marshal(ctxVars)
|
|
||||||
return &csi.CreateVolumeResponse{
|
|
||||||
Volume: &csi.Volume{
|
|
||||||
// Ugly, but VolumeContext isn't passed to DeleteVolume :-(
|
|
||||||
VolumeId: string(volumeIdJson),
|
|
||||||
CapacityBytes: volSize,
|
|
||||||
},
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// DeleteVolume deletes the given volume
|
|
||||||
func (cs *ControllerServer) DeleteVolume(ctx context.Context, req *csi.DeleteVolumeRequest) (*csi.DeleteVolumeResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received controller delete volume request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
if (req == nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "request cannot be empty")
|
|
||||||
}
|
|
||||||
|
|
||||||
ctxVars := make(map[string]string)
|
|
||||||
err := json.Unmarshal([]byte(req.VolumeId), &ctxVars)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "volume ID not in JSON format")
|
|
||||||
}
|
|
||||||
volName := ctxVars["name"]
|
|
||||||
|
|
||||||
_, etcdUrl, etcdPrefix := GetConnectionParams(ctxVars)
|
|
||||||
if (len(etcdUrl) == 0)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "no etcdUrl in storage class configuration and no etcd_address in vitastor.conf")
|
|
||||||
}
|
|
||||||
|
|
||||||
cli, err := clientv3.New(clientv3.Config{
|
|
||||||
DialTimeout: ETCD_TIMEOUT,
|
|
||||||
Endpoints: etcdUrl,
|
|
||||||
})
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to connect to etcd at "+strings.Join(etcdUrl, ",")+": "+err.Error())
|
|
||||||
}
|
|
||||||
defer cli.Close()
|
|
||||||
|
|
||||||
// Find inode by name
|
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), ETCD_TIMEOUT)
|
|
||||||
resp, err := cli.Get(ctx, etcdPrefix+"/index/image/"+volName)
|
|
||||||
cancel()
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to read key from etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
if (len(resp.Kvs) == 0)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.NotFound, "volume "+volName+" does not exist")
|
|
||||||
}
|
|
||||||
var idx InodeIndex
|
|
||||||
err = json.Unmarshal(resp.Kvs[0].Value, &idx)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "invalid /index/image/"+volName+" key in etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get inode config
|
|
||||||
inodeCfgKey := fmt.Sprintf("%s/config/inode/%d/%d", etcdPrefix, idx.PoolId, idx.Id)
|
|
||||||
ctx, cancel = context.WithTimeout(context.Background(), ETCD_TIMEOUT)
|
|
||||||
resp, err = cli.Get(ctx, inodeCfgKey)
|
|
||||||
cancel()
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to read key from etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
if (len(resp.Kvs) == 0)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.NotFound, "volume "+volName+" does not exist")
|
|
||||||
}
|
|
||||||
var inodeCfg InodeConfig
|
|
||||||
err = json.Unmarshal(resp.Kvs[0].Value, &inodeCfg)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "invalid "+inodeCfgKey+" key in etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete inode data by invoking vitastor-cli
|
|
||||||
args := []string{
|
|
||||||
"rm-data", "--etcd_address", strings.Join(etcdUrl, ","),
|
|
||||||
"--pool", fmt.Sprintf("%d", idx.PoolId),
|
|
||||||
"--inode", fmt.Sprintf("%d", idx.Id),
|
|
||||||
}
|
|
||||||
if (ctxVars["configPath"] != "")
|
|
||||||
{
|
|
||||||
args = append(args, "--config_path", ctxVars["configPath"])
|
|
||||||
}
|
|
||||||
c := exec.Command("/usr/bin/vitastor-cli", args...)
|
|
||||||
var stderr bytes.Buffer
|
|
||||||
c.Stdout = nil
|
|
||||||
c.Stderr = &stderr
|
|
||||||
err = c.Run()
|
|
||||||
stderrStr := string(stderr.Bytes())
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("vitastor-cli rm-data failed: %s, status %s\n", stderrStr, err)
|
|
||||||
return nil, status.Error(codes.Internal, stderrStr+" (status "+err.Error()+")")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete inode config in etcd
|
|
||||||
ctx, cancel = context.WithTimeout(context.Background(), ETCD_TIMEOUT)
|
|
||||||
txnResp, err := cli.Txn(ctx).Then(
|
|
||||||
clientv3.OpDelete(fmt.Sprintf("%s/index/image/%s", etcdPrefix, volName)),
|
|
||||||
clientv3.OpDelete(fmt.Sprintf("%s/config/inode/%d/%d", etcdPrefix, idx.PoolId, idx.Id)),
|
|
||||||
).Commit()
|
|
||||||
cancel()
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to delete keys in etcd: "+err.Error())
|
|
||||||
}
|
|
||||||
if (!txnResp.Succeeded)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "failed to delete keys in etcd: transaction failed")
|
|
||||||
}
|
|
||||||
|
|
||||||
return &csi.DeleteVolumeResponse{}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ControllerPublishVolume return Unimplemented error
|
|
||||||
func (cs *ControllerServer) ControllerPublishVolume(ctx context.Context, req *csi.ControllerPublishVolumeRequest) (*csi.ControllerPublishVolumeResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ControllerUnpublishVolume return Unimplemented error
|
|
||||||
func (cs *ControllerServer) ControllerUnpublishVolume(ctx context.Context, req *csi.ControllerUnpublishVolumeRequest) (*csi.ControllerUnpublishVolumeResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ValidateVolumeCapabilities checks whether the volume capabilities requested are supported.
|
|
||||||
func (cs *ControllerServer) ValidateVolumeCapabilities(ctx context.Context, req *csi.ValidateVolumeCapabilitiesRequest) (*csi.ValidateVolumeCapabilitiesResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received controller validate volume capability request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
if (req == nil)
|
|
||||||
{
|
|
||||||
return nil, status.Errorf(codes.InvalidArgument, "request is nil")
|
|
||||||
}
|
|
||||||
volumeID := req.GetVolumeId()
|
|
||||||
if (volumeID == "")
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "volumeId is nil")
|
|
||||||
}
|
|
||||||
volumeCapabilities := req.GetVolumeCapabilities()
|
|
||||||
if (volumeCapabilities == nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "volumeCapabilities is nil")
|
|
||||||
}
|
|
||||||
|
|
||||||
var volumeCapabilityAccessModes []*csi.VolumeCapability_AccessMode
|
|
||||||
for _, mode := range []csi.VolumeCapability_AccessMode_Mode{
|
|
||||||
csi.VolumeCapability_AccessMode_SINGLE_NODE_WRITER,
|
|
||||||
csi.VolumeCapability_AccessMode_MULTI_NODE_MULTI_WRITER,
|
|
||||||
} {
|
|
||||||
volumeCapabilityAccessModes = append(volumeCapabilityAccessModes, &csi.VolumeCapability_AccessMode{Mode: mode})
|
|
||||||
}
|
|
||||||
|
|
||||||
capabilitySupport := false
|
|
||||||
for _, capability := range volumeCapabilities
|
|
||||||
{
|
|
||||||
for _, volumeCapabilityAccessMode := range volumeCapabilityAccessModes
|
|
||||||
{
|
|
||||||
if (volumeCapabilityAccessMode.Mode == capability.AccessMode.Mode)
|
|
||||||
{
|
|
||||||
capabilitySupport = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!capabilitySupport)
|
|
||||||
{
|
|
||||||
return nil, status.Errorf(codes.NotFound, "%v not supported", req.GetVolumeCapabilities())
|
|
||||||
}
|
|
||||||
|
|
||||||
return &csi.ValidateVolumeCapabilitiesResponse{
|
|
||||||
Confirmed: &csi.ValidateVolumeCapabilitiesResponse_Confirmed{
|
|
||||||
VolumeCapabilities: req.VolumeCapabilities,
|
|
||||||
},
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListVolumes returns a list of volumes
|
|
||||||
func (cs *ControllerServer) ListVolumes(ctx context.Context, req *csi.ListVolumesRequest) (*csi.ListVolumesResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetCapacity returns the capacity of the storage pool
|
|
||||||
func (cs *ControllerServer) GetCapacity(ctx context.Context, req *csi.GetCapacityRequest) (*csi.GetCapacityResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ControllerGetCapabilities returns the capabilities of the controller service.
|
|
||||||
func (cs *ControllerServer) ControllerGetCapabilities(ctx context.Context, req *csi.ControllerGetCapabilitiesRequest) (*csi.ControllerGetCapabilitiesResponse, error)
|
|
||||||
{
|
|
||||||
functionControllerServerCapabilities := func(cap csi.ControllerServiceCapability_RPC_Type) *csi.ControllerServiceCapability
|
|
||||||
{
|
|
||||||
return &csi.ControllerServiceCapability{
|
|
||||||
Type: &csi.ControllerServiceCapability_Rpc{
|
|
||||||
Rpc: &csi.ControllerServiceCapability_RPC{
|
|
||||||
Type: cap,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var controllerServerCapabilities []*csi.ControllerServiceCapability
|
|
||||||
for _, capability := range []csi.ControllerServiceCapability_RPC_Type{
|
|
||||||
csi.ControllerServiceCapability_RPC_CREATE_DELETE_VOLUME,
|
|
||||||
csi.ControllerServiceCapability_RPC_LIST_VOLUMES,
|
|
||||||
csi.ControllerServiceCapability_RPC_EXPAND_VOLUME,
|
|
||||||
csi.ControllerServiceCapability_RPC_CREATE_DELETE_SNAPSHOT,
|
|
||||||
} {
|
|
||||||
controllerServerCapabilities = append(controllerServerCapabilities, functionControllerServerCapabilities(capability))
|
|
||||||
}
|
|
||||||
|
|
||||||
return &csi.ControllerGetCapabilitiesResponse{
|
|
||||||
Capabilities: controllerServerCapabilities,
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// CreateSnapshot create snapshot of an existing PV
|
|
||||||
func (cs *ControllerServer) CreateSnapshot(ctx context.Context, req *csi.CreateSnapshotRequest) (*csi.CreateSnapshotResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// DeleteSnapshot delete provided snapshot of a PV
|
|
||||||
func (cs *ControllerServer) DeleteSnapshot(ctx context.Context, req *csi.DeleteSnapshotRequest) (*csi.DeleteSnapshotResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ListSnapshots list the snapshots of a PV
|
|
||||||
func (cs *ControllerServer) ListSnapshots(ctx context.Context, req *csi.ListSnapshotsRequest) (*csi.ListSnapshotsResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ControllerExpandVolume resizes a volume
|
|
||||||
func (cs *ControllerServer) ControllerExpandVolume(ctx context.Context, req *csi.ControllerExpandVolumeRequest) (*csi.ControllerExpandVolumeResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// ControllerGetVolume get volume info
|
|
||||||
func (cs *ControllerServer) ControllerGetVolume(ctx context.Context, req *csi.ControllerGetVolumeRequest) (*csi.ControllerGetVolumeResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
137
csi/src/grpc.go
137
csi/src/grpc.go
|
@ -1,137 +0,0 @@
|
||||||
/*
|
|
||||||
Copyright 2017 The Kubernetes Authors.
|
|
||||||
|
|
||||||
Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
you may not use this file except in compliance with the License.
|
|
||||||
You may obtain a copy of the License at
|
|
||||||
|
|
||||||
http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
|
|
||||||
Unless required by applicable law or agreed to in writing, software
|
|
||||||
distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
See the License for the specific language governing permissions and
|
|
||||||
limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
package vitastor
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
|
||||||
|
|
||||||
"github.com/golang/glog"
|
|
||||||
"golang.org/x/net/context"
|
|
||||||
"google.golang.org/grpc"
|
|
||||||
|
|
||||||
"github.com/container-storage-interface/spec/lib/go/csi"
|
|
||||||
"github.com/kubernetes-csi/csi-lib-utils/protosanitizer"
|
|
||||||
)
|
|
||||||
|
|
||||||
// Defines Non blocking GRPC server interfaces
|
|
||||||
type NonBlockingGRPCServer interface {
|
|
||||||
// Start services at the endpoint
|
|
||||||
Start(endpoint string, ids csi.IdentityServer, cs csi.ControllerServer, ns csi.NodeServer)
|
|
||||||
// Waits for the service to stop
|
|
||||||
Wait()
|
|
||||||
// Stops the service gracefully
|
|
||||||
Stop()
|
|
||||||
// Stops the service forcefully
|
|
||||||
ForceStop()
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewNonBlockingGRPCServer() NonBlockingGRPCServer {
|
|
||||||
return &nonBlockingGRPCServer{}
|
|
||||||
}
|
|
||||||
|
|
||||||
// NonBlocking server
|
|
||||||
type nonBlockingGRPCServer struct {
|
|
||||||
wg sync.WaitGroup
|
|
||||||
server *grpc.Server
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *nonBlockingGRPCServer) Start(endpoint string, ids csi.IdentityServer, cs csi.ControllerServer, ns csi.NodeServer) {
|
|
||||||
|
|
||||||
s.wg.Add(1)
|
|
||||||
|
|
||||||
go s.serve(endpoint, ids, cs, ns)
|
|
||||||
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *nonBlockingGRPCServer) Wait() {
|
|
||||||
s.wg.Wait()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *nonBlockingGRPCServer) Stop() {
|
|
||||||
s.server.GracefulStop()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *nonBlockingGRPCServer) ForceStop() {
|
|
||||||
s.server.Stop()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *nonBlockingGRPCServer) serve(endpoint string, ids csi.IdentityServer, cs csi.ControllerServer, ns csi.NodeServer) {
|
|
||||||
|
|
||||||
proto, addr, err := ParseEndpoint(endpoint)
|
|
||||||
if err != nil {
|
|
||||||
glog.Fatal(err.Error())
|
|
||||||
}
|
|
||||||
|
|
||||||
if proto == "unix" {
|
|
||||||
addr = "/" + addr
|
|
||||||
if err := os.Remove(addr); err != nil && !os.IsNotExist(err) {
|
|
||||||
glog.Fatalf("Failed to remove %s, error: %s", addr, err.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
listener, err := net.Listen(proto, addr)
|
|
||||||
if err != nil {
|
|
||||||
glog.Fatalf("Failed to listen: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
opts := []grpc.ServerOption{
|
|
||||||
grpc.UnaryInterceptor(logGRPC),
|
|
||||||
}
|
|
||||||
server := grpc.NewServer(opts...)
|
|
||||||
s.server = server
|
|
||||||
|
|
||||||
if ids != nil {
|
|
||||||
csi.RegisterIdentityServer(server, ids)
|
|
||||||
}
|
|
||||||
if cs != nil {
|
|
||||||
csi.RegisterControllerServer(server, cs)
|
|
||||||
}
|
|
||||||
if ns != nil {
|
|
||||||
csi.RegisterNodeServer(server, ns)
|
|
||||||
}
|
|
||||||
|
|
||||||
glog.Infof("Listening for connections on address: %#v", listener.Addr())
|
|
||||||
|
|
||||||
server.Serve(listener)
|
|
||||||
}
|
|
||||||
|
|
||||||
func ParseEndpoint(ep string) (string, string, error) {
|
|
||||||
if strings.HasPrefix(strings.ToLower(ep), "unix://") || strings.HasPrefix(strings.ToLower(ep), "tcp://") {
|
|
||||||
s := strings.SplitN(ep, "://", 2)
|
|
||||||
if s[1] != "" {
|
|
||||||
return s[0], s[1], nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return "", "", fmt.Errorf("Invalid endpoint: %v", ep)
|
|
||||||
}
|
|
||||||
|
|
||||||
func logGRPC(ctx context.Context, req interface{}, info *grpc.UnaryServerInfo, handler grpc.UnaryHandler) (interface{}, error) {
|
|
||||||
glog.V(3).Infof("GRPC call: %s", info.FullMethod)
|
|
||||||
glog.V(5).Infof("GRPC request: %s", protosanitizer.StripSecrets(req))
|
|
||||||
resp, err := handler(ctx, req)
|
|
||||||
if err != nil {
|
|
||||||
glog.Errorf("GRPC error: %v", err)
|
|
||||||
} else {
|
|
||||||
glog.V(5).Infof("GRPC response: %s", protosanitizer.StripSecrets(resp))
|
|
||||||
}
|
|
||||||
return resp, err
|
|
||||||
}
|
|
|
@ -1,60 +0,0 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
|
||||||
// License: VNPL-1.1 or GNU GPL-2.0+ (see README.md for details)
|
|
||||||
|
|
||||||
package vitastor
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
|
|
||||||
"github.com/kubernetes-csi/csi-lib-utils/protosanitizer"
|
|
||||||
"k8s.io/klog"
|
|
||||||
|
|
||||||
"github.com/container-storage-interface/spec/lib/go/csi"
|
|
||||||
)
|
|
||||||
|
|
||||||
// IdentityServer struct of Vitastor CSI driver with supported methods of CSI identity server spec.
|
|
||||||
type IdentityServer struct
|
|
||||||
{
|
|
||||||
*Driver
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewIdentityServer create new instance identity
|
|
||||||
func NewIdentityServer(driver *Driver) *IdentityServer
|
|
||||||
{
|
|
||||||
return &IdentityServer{
|
|
||||||
Driver: driver,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetPluginInfo returns metadata of the plugin
|
|
||||||
func (is *IdentityServer) GetPluginInfo(ctx context.Context, req *csi.GetPluginInfoRequest) (*csi.GetPluginInfoResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received identity plugin info request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
return &csi.GetPluginInfoResponse{
|
|
||||||
Name: vitastorCSIDriverName,
|
|
||||||
VendorVersion: vitastorCSIDriverVersion,
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// GetPluginCapabilities returns available capabilities of the plugin
|
|
||||||
func (is *IdentityServer) GetPluginCapabilities(ctx context.Context, req *csi.GetPluginCapabilitiesRequest) (*csi.GetPluginCapabilitiesResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received identity plugin capabilities request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
return &csi.GetPluginCapabilitiesResponse{
|
|
||||||
Capabilities: []*csi.PluginCapability{
|
|
||||||
{
|
|
||||||
Type: &csi.PluginCapability_Service_{
|
|
||||||
Service: &csi.PluginCapability_Service{
|
|
||||||
Type: csi.PluginCapability_Service_CONTROLLER_SERVICE,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Probe returns the health and readiness of the plugin
|
|
||||||
func (is *IdentityServer) Probe(ctx context.Context, req *csi.ProbeRequest) (*csi.ProbeResponse, error)
|
|
||||||
{
|
|
||||||
return &csi.ProbeResponse{}, nil
|
|
||||||
}
|
|
|
@ -1,279 +0,0 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
|
||||||
// License: VNPL-1.1 or GNU GPL-2.0+ (see README.md for details)
|
|
||||||
|
|
||||||
package vitastor
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"os"
|
|
||||||
"os/exec"
|
|
||||||
"encoding/json"
|
|
||||||
"strings"
|
|
||||||
"bytes"
|
|
||||||
|
|
||||||
"google.golang.org/grpc/codes"
|
|
||||||
"google.golang.org/grpc/status"
|
|
||||||
"k8s.io/utils/mount"
|
|
||||||
utilexec "k8s.io/utils/exec"
|
|
||||||
|
|
||||||
"github.com/container-storage-interface/spec/lib/go/csi"
|
|
||||||
"github.com/kubernetes-csi/csi-lib-utils/protosanitizer"
|
|
||||||
"k8s.io/klog"
|
|
||||||
)
|
|
||||||
|
|
||||||
// NodeServer struct of Vitastor CSI driver with supported methods of CSI node server spec.
|
|
||||||
type NodeServer struct
|
|
||||||
{
|
|
||||||
*Driver
|
|
||||||
mounter mount.Interface
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewNodeServer create new instance node
|
|
||||||
func NewNodeServer(driver *Driver) *NodeServer
|
|
||||||
{
|
|
||||||
return &NodeServer{
|
|
||||||
Driver: driver,
|
|
||||||
mounter: mount.New(""),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeStageVolume mounts the volume to a staging path on the node.
|
|
||||||
func (ns *NodeServer) NodeStageVolume(ctx context.Context, req *csi.NodeStageVolumeRequest) (*csi.NodeStageVolumeResponse, error)
|
|
||||||
{
|
|
||||||
return &csi.NodeStageVolumeResponse{}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeUnstageVolume unstages the volume from the staging path
|
|
||||||
func (ns *NodeServer) NodeUnstageVolume(ctx context.Context, req *csi.NodeUnstageVolumeRequest) (*csi.NodeUnstageVolumeResponse, error)
|
|
||||||
{
|
|
||||||
return &csi.NodeUnstageVolumeResponse{}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func Contains(list []string, s string) bool
|
|
||||||
{
|
|
||||||
for i := 0; i < len(list); i++
|
|
||||||
{
|
|
||||||
if (list[i] == s)
|
|
||||||
{
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodePublishVolume mounts the volume mounted to the staging path to the target path
|
|
||||||
func (ns *NodeServer) NodePublishVolume(ctx context.Context, req *csi.NodePublishVolumeRequest) (*csi.NodePublishVolumeResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received node publish volume request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
|
|
||||||
targetPath := req.GetTargetPath()
|
|
||||||
|
|
||||||
// Check that it's not already mounted
|
|
||||||
free, error := mount.IsNotMountPoint(ns.mounter, targetPath)
|
|
||||||
if (error != nil)
|
|
||||||
{
|
|
||||||
if (os.IsNotExist(error))
|
|
||||||
{
|
|
||||||
error := os.MkdirAll(targetPath, 0777)
|
|
||||||
if (error != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, error.Error())
|
|
||||||
}
|
|
||||||
free = true
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, error.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!free)
|
|
||||||
{
|
|
||||||
return &csi.NodePublishVolumeResponse{}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
ctxVars := make(map[string]string)
|
|
||||||
err := json.Unmarshal([]byte(req.VolumeId), &ctxVars)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, "volume ID not in JSON format")
|
|
||||||
}
|
|
||||||
volName := ctxVars["name"]
|
|
||||||
|
|
||||||
_, etcdUrl, etcdPrefix := GetConnectionParams(ctxVars)
|
|
||||||
if (len(etcdUrl) == 0)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.InvalidArgument, "no etcdUrl in storage class configuration and no etcd_address in vitastor.conf")
|
|
||||||
}
|
|
||||||
|
|
||||||
// Map NBD device
|
|
||||||
// FIXME: Check if already mapped
|
|
||||||
args := []string{
|
|
||||||
"map", "--etcd_address", strings.Join(etcdUrl, ","),
|
|
||||||
"--etcd_prefix", etcdPrefix,
|
|
||||||
"--image", volName,
|
|
||||||
};
|
|
||||||
if (ctxVars["configPath"] != "")
|
|
||||||
{
|
|
||||||
args = append(args, "--config_path", ctxVars["configPath"])
|
|
||||||
}
|
|
||||||
if (req.GetReadonly())
|
|
||||||
{
|
|
||||||
args = append(args, "--readonly", "1")
|
|
||||||
}
|
|
||||||
c := exec.Command("/usr/bin/vitastor-nbd", args...)
|
|
||||||
var stdout, stderr bytes.Buffer
|
|
||||||
c.Stdout, c.Stderr = &stdout, &stderr
|
|
||||||
err = c.Run()
|
|
||||||
stdoutStr, stderrStr := string(stdout.Bytes()), string(stderr.Bytes())
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("vitastor-nbd map failed: %s, status %s\n", stdoutStr+stderrStr, err)
|
|
||||||
return nil, status.Error(codes.Internal, stdoutStr+stderrStr+" (status "+err.Error()+")")
|
|
||||||
}
|
|
||||||
devicePath := strings.TrimSpace(stdoutStr)
|
|
||||||
|
|
||||||
// Check existing format
|
|
||||||
diskMounter := &mount.SafeFormatAndMount{Interface: ns.mounter, Exec: utilexec.New()}
|
|
||||||
existingFormat, err := diskMounter.GetDiskFormat(devicePath)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("failed to get disk format for path %s, error: %v", err)
|
|
||||||
// unmap NBD device
|
|
||||||
unmapOut, unmapErr := exec.Command("/usr/bin/vitastor-nbd", "unmap", devicePath).CombinedOutput()
|
|
||||||
if (unmapErr != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("failed to unmap NBD device %s: %s, error: %v", devicePath, unmapOut, unmapErr)
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
// Format the device (ext4 or xfs)
|
|
||||||
fsType := req.GetVolumeCapability().GetMount().GetFsType()
|
|
||||||
isBlock := req.GetVolumeCapability().GetBlock() != nil
|
|
||||||
opt := req.GetVolumeCapability().GetMount().GetMountFlags()
|
|
||||||
opt = append(opt, "_netdev")
|
|
||||||
if ((req.VolumeCapability.AccessMode.Mode == csi.VolumeCapability_AccessMode_MULTI_NODE_READER_ONLY ||
|
|
||||||
req.VolumeCapability.AccessMode.Mode == csi.VolumeCapability_AccessMode_SINGLE_NODE_READER_ONLY) &&
|
|
||||||
!Contains(opt, "ro"))
|
|
||||||
{
|
|
||||||
opt = append(opt, "ro")
|
|
||||||
}
|
|
||||||
if (fsType == "xfs")
|
|
||||||
{
|
|
||||||
opt = append(opt, "nouuid")
|
|
||||||
}
|
|
||||||
readOnly := Contains(opt, "ro")
|
|
||||||
if (existingFormat == "" && !readOnly)
|
|
||||||
{
|
|
||||||
args := []string{}
|
|
||||||
switch fsType
|
|
||||||
{
|
|
||||||
case "ext4":
|
|
||||||
args = []string{"-m0", "-Enodiscard,lazy_itable_init=1,lazy_journal_init=1", devicePath}
|
|
||||||
case "xfs":
|
|
||||||
args = []string{"-K", devicePath}
|
|
||||||
}
|
|
||||||
if (len(args) > 0)
|
|
||||||
{
|
|
||||||
cmdOut, cmdErr := diskMounter.Exec.Command("mkfs."+fsType, args...).CombinedOutput()
|
|
||||||
if (cmdErr != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("failed to run mkfs error: %v, output: %v", cmdErr, string(cmdOut))
|
|
||||||
// unmap NBD device
|
|
||||||
unmapOut, unmapErr := exec.Command("/usr/bin/vitastor-nbd", "unmap", devicePath).CombinedOutput()
|
|
||||||
if (unmapErr != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("failed to unmap NBD device %s: %s, error: %v", devicePath, unmapOut, unmapErr)
|
|
||||||
}
|
|
||||||
return nil, status.Error(codes.Internal, cmdErr.Error())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (isBlock)
|
|
||||||
{
|
|
||||||
opt = append(opt, "bind")
|
|
||||||
err = diskMounter.Mount(devicePath, targetPath, fsType, opt)
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
err = diskMounter.FormatAndMount(devicePath, targetPath, fsType, opt)
|
|
||||||
}
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf(
|
|
||||||
"failed to mount device path (%s) to path (%s) for volume (%s) error: %s",
|
|
||||||
devicePath, targetPath, volName, err,
|
|
||||||
)
|
|
||||||
// unmap NBD device
|
|
||||||
unmapOut, unmapErr := exec.Command("/usr/bin/vitastor-nbd", "unmap", devicePath).CombinedOutput()
|
|
||||||
if (unmapErr != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("failed to unmap NBD device %s: %s, error: %v", devicePath, unmapOut, unmapErr)
|
|
||||||
}
|
|
||||||
return nil, status.Error(codes.Internal, err.Error())
|
|
||||||
}
|
|
||||||
return &csi.NodePublishVolumeResponse{}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeUnpublishVolume unmounts the volume from the target path
|
|
||||||
func (ns *NodeServer) NodeUnpublishVolume(ctx context.Context, req *csi.NodeUnpublishVolumeRequest) (*csi.NodeUnpublishVolumeResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received node unpublish volume request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
targetPath := req.GetTargetPath()
|
|
||||||
devicePath, refCount, err := mount.GetDeviceNameFromMount(ns.mounter, targetPath)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
if (os.IsNotExist(err))
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.NotFound, "Target path not found")
|
|
||||||
}
|
|
||||||
return nil, status.Error(codes.Internal, err.Error())
|
|
||||||
}
|
|
||||||
if (devicePath == "")
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.NotFound, "Volume not mounted")
|
|
||||||
}
|
|
||||||
// unmount
|
|
||||||
err = mount.CleanupMountPoint(targetPath, ns.mounter, false)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Internal, err.Error())
|
|
||||||
}
|
|
||||||
// unmap NBD device
|
|
||||||
if (refCount == 1)
|
|
||||||
{
|
|
||||||
unmapOut, unmapErr := exec.Command("/usr/bin/vitastor-nbd", "unmap", devicePath).CombinedOutput()
|
|
||||||
if (unmapErr != nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("failed to unmap NBD device %s: %s, error: %v", devicePath, unmapOut, unmapErr)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return &csi.NodeUnpublishVolumeResponse{}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeGetVolumeStats returns volume capacity statistics available for the volume
|
|
||||||
func (ns *NodeServer) NodeGetVolumeStats(ctx context.Context, req *csi.NodeGetVolumeStatsRequest) (*csi.NodeGetVolumeStatsResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeExpandVolume expanding the file system on the node
|
|
||||||
func (ns *NodeServer) NodeExpandVolume(ctx context.Context, req *csi.NodeExpandVolumeRequest) (*csi.NodeExpandVolumeResponse, error)
|
|
||||||
{
|
|
||||||
return nil, status.Error(codes.Unimplemented, "")
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeGetCapabilities returns the supported capabilities of the node server
|
|
||||||
func (ns *NodeServer) NodeGetCapabilities(ctx context.Context, req *csi.NodeGetCapabilitiesRequest) (*csi.NodeGetCapabilitiesResponse, error)
|
|
||||||
{
|
|
||||||
return &csi.NodeGetCapabilitiesResponse{}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// NodeGetInfo returns NodeGetInfoResponse for CO.
|
|
||||||
func (ns *NodeServer) NodeGetInfo(ctx context.Context, req *csi.NodeGetInfoRequest) (*csi.NodeGetInfoResponse, error)
|
|
||||||
{
|
|
||||||
klog.Infof("received node get info request %+v", protosanitizer.StripSecrets(req))
|
|
||||||
return &csi.NodeGetInfoResponse{
|
|
||||||
NodeId: ns.NodeID,
|
|
||||||
}, nil
|
|
||||||
}
|
|
|
@ -1,36 +0,0 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
|
||||||
// License: VNPL-1.1 or GNU GPL-2.0+ (see README.md for details)
|
|
||||||
|
|
||||||
package vitastor
|
|
||||||
|
|
||||||
import (
|
|
||||||
"k8s.io/klog"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Driver struct
|
|
||||||
{
|
|
||||||
*Config
|
|
||||||
}
|
|
||||||
|
|
||||||
// NewDriver create new instance driver
|
|
||||||
func NewDriver(config *Config) (*Driver, error)
|
|
||||||
{
|
|
||||||
if (config == nil)
|
|
||||||
{
|
|
||||||
klog.Errorf("Vitastor CSI driver initialization failed")
|
|
||||||
return nil, nil
|
|
||||||
}
|
|
||||||
driver := &Driver{
|
|
||||||
Config: config,
|
|
||||||
}
|
|
||||||
klog.Infof("Vitastor CSI driver initialized")
|
|
||||||
return driver, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start server
|
|
||||||
func (driver *Driver) Run()
|
|
||||||
{
|
|
||||||
server := NewNonBlockingGRPCServer()
|
|
||||||
server.Start(driver.Endpoint, NewIdentityServer(driver), NewControllerServer(driver), NewNodeServer(driver))
|
|
||||||
server.Wait()
|
|
||||||
}
|
|
|
@ -1,39 +0,0 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
|
||||||
// License: VNPL-1.1 or GNU GPL-2.0+ (see README.md for details)
|
|
||||||
|
|
||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
"k8s.io/klog"
|
|
||||||
"vitastor.io/csi/src"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main()
|
|
||||||
{
|
|
||||||
var config = vitastor.NewConfig()
|
|
||||||
flag.StringVar(&config.Endpoint, "endpoint", "", "CSI endpoint")
|
|
||||||
flag.StringVar(&config.NodeID, "node", "", "Node ID")
|
|
||||||
flag.Parse()
|
|
||||||
if (config.Endpoint == "")
|
|
||||||
{
|
|
||||||
config.Endpoint = os.Getenv("CSI_ENDPOINT")
|
|
||||||
}
|
|
||||||
if (config.NodeID == "")
|
|
||||||
{
|
|
||||||
config.NodeID = os.Getenv("NODE_ID")
|
|
||||||
}
|
|
||||||
if (config.Endpoint == "" && config.NodeID == "")
|
|
||||||
{
|
|
||||||
fmt.Fprintf(os.Stderr, "Please set -endpoint and -node / CSI_ENDPOINT & NODE_ID env vars\n")
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
drv, err := vitastor.NewDriver(config)
|
|
||||||
if (err != nil)
|
|
||||||
{
|
|
||||||
klog.Fatalln(err)
|
|
||||||
}
|
|
||||||
drv.Run()
|
|
||||||
}
|
|
|
@ -1,7 +1,7 @@
|
||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
cat < vitastor.Dockerfile > ../Dockerfile
|
sed 's/$REL/bullseye/g' < vitastor.Dockerfile > ../Dockerfile
|
||||||
cd ..
|
cd ..
|
||||||
mkdir -p packages
|
mkdir -p packages
|
||||||
sudo podman build --build-arg REL=bullseye -v `pwd`/packages:/root/packages -f Dockerfile .
|
sudo podman build -v `pwd`/packages:/root/packages -f Dockerfile .
|
||||||
rm Dockerfile
|
rm Dockerfile
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
cat < vitastor.Dockerfile > ../Dockerfile
|
sed 's/$REL/buster/g' < vitastor.Dockerfile > ../Dockerfile
|
||||||
cd ..
|
cd ..
|
||||||
mkdir -p packages
|
mkdir -p packages
|
||||||
sudo podman build --build-arg REL=buster -v `pwd`/packages:/root/packages -f Dockerfile .
|
sudo podman build -v `pwd`/packages:/root/packages -f Dockerfile .
|
||||||
rm Dockerfile
|
rm Dockerfile
|
||||||
|
|
|
@ -1,18 +1,8 @@
|
||||||
vitastor (0.6.10-1) unstable; urgency=medium
|
vitastor (0.6.2-1) unstable; urgency=medium
|
||||||
|
|
||||||
* RDMA support
|
|
||||||
* Bugfixes
|
* Bugfixes
|
||||||
|
|
||||||
-- Vitaliy Filippov <vitalif@yourcmc.ru> Sat, 01 May 2021 18:46:10 +0300
|
-- Vitaliy Filippov <vitalif@yourcmc.ru> Tue, 02 Feb 2021 23:01:24 +0300
|
||||||
|
|
||||||
vitastor (0.6.0-1) unstable; urgency=medium
|
|
||||||
|
|
||||||
* Snapshots and Copy-on-Write clones
|
|
||||||
* Image metadata in etcd (name, size)
|
|
||||||
* Image I/O and space statistics in etcd
|
|
||||||
* Write throttling for smoothing random write workloads in SSD+HDD configurations
|
|
||||||
|
|
||||||
-- Vitaliy Filippov <vitalif@yourcmc.ru> Sun, 11 Apr 2021 00:49:18 +0300
|
|
||||||
|
|
||||||
vitastor (0.5.1-1) unstable; urgency=medium
|
vitastor (0.5.1-1) unstable; urgency=medium
|
||||||
|
|
||||||
|
|
|
@ -2,54 +2,16 @@ Source: vitastor
|
||||||
Section: admin
|
Section: admin
|
||||||
Priority: optional
|
Priority: optional
|
||||||
Maintainer: Vitaliy Filippov <vitalif@yourcmc.ru>
|
Maintainer: Vitaliy Filippov <vitalif@yourcmc.ru>
|
||||||
Build-Depends: debhelper, liburing-dev (>= 0.6), g++ (>= 8), libstdc++6 (>= 8), linux-libc-dev, libgoogle-perftools-dev, libjerasure-dev, libgf-complete-dev, libibverbs-dev
|
Build-Depends: debhelper, liburing-dev (>= 0.6), g++ (>= 8), libstdc++6 (>= 8), linux-libc-dev, libgoogle-perftools-dev, libjerasure-dev, libgf-complete-dev
|
||||||
Standards-Version: 4.5.0
|
Standards-Version: 4.5.0
|
||||||
Homepage: https://vitastor.io/
|
Homepage: https://vitastor.io/
|
||||||
Rules-Requires-Root: no
|
Rules-Requires-Root: no
|
||||||
|
|
||||||
Package: vitastor
|
Package: vitastor
|
||||||
Architecture: amd64
|
Architecture: amd64
|
||||||
Depends: vitastor-osd, vitastor-mon, vitastor-client, vitastor-client-dev, vitastor-fio
|
Depends: ${shlibs:Depends}, ${misc:Depends}, fio (= ${dep:fio}), qemu (= ${dep:qemu}), nodejs (>= 10), node-sprintf-js, node-ws (>= 7), libjerasure2, lp-solve
|
||||||
Description: Vitastor, a fast software-defined clustered block storage
|
Description: Vitastor, a fast software-defined clustered block storage
|
||||||
Vitastor is a small, simple and fast clustered block storage (storage for VM drives),
|
Vitastor is a small, simple and fast clustered block storage (storage for VM drives),
|
||||||
architecturally similar to Ceph which means strong consistency, primary-replication,
|
architecturally similar to Ceph which means strong consistency, primary-replication,
|
||||||
symmetric clustering and automatic data distribution over any number of drives of any
|
symmetric clustering and automatic data distribution over any number of drives of any
|
||||||
size with configurable redundancy (replication or erasure codes/XOR).
|
size with configurable redundancy (replication or erasure codes/XOR).
|
||||||
|
|
||||||
Package: vitastor-osd
|
|
||||||
Architecture: amd64
|
|
||||||
Depends: ${shlibs:Depends}, ${misc:Depends}, vitastor-client (= ${binary:Version})
|
|
||||||
Description: Vitastor, a fast software-defined clustered block storage - object storage daemon
|
|
||||||
Vitastor object storage daemon, i.e. server program that stores data.
|
|
||||||
|
|
||||||
Package: vitastor-mon
|
|
||||||
Architecture: amd64
|
|
||||||
Depends: ${misc:Depends}, nodejs (>= 10), node-sprintf-js, node-ws (>= 7), lp-solve
|
|
||||||
Description: Vitastor, a fast software-defined clustered block storage - monitor
|
|
||||||
Vitastor monitor, i.e. server program responsible for watching cluster state and
|
|
||||||
scheduling cluster-level operations.
|
|
||||||
|
|
||||||
Package: vitastor-client
|
|
||||||
Architecture: amd64
|
|
||||||
Depends: ${shlibs:Depends}, ${misc:Depends}
|
|
||||||
Description: Vitastor, a fast software-defined clustered block storage - client
|
|
||||||
Vitastor client library and command-line interface.
|
|
||||||
|
|
||||||
Package: vitastor-client-dev
|
|
||||||
Section: devel
|
|
||||||
Architecture: amd64
|
|
||||||
Depends: ${misc:Depends}, vitastor-client (= ${binary:Version})
|
|
||||||
Description: Vitastor, a fast software-defined clustered block storage - development files
|
|
||||||
Vitastor library headers for development.
|
|
||||||
|
|
||||||
Package: vitastor-fio
|
|
||||||
Architecture: amd64
|
|
||||||
Depends: ${shlibs:Depends}, ${misc:Depends}, vitastor-client (= ${binary:Version}), fio (= ${dep:fio})
|
|
||||||
Description: Vitastor, a fast software-defined clustered block storage - fio drivers
|
|
||||||
Vitastor fio drivers for benchmarking.
|
|
||||||
|
|
||||||
Package: pve-storage-vitastor
|
|
||||||
Architecture: amd64
|
|
||||||
Depends: ${shlibs:Depends}, ${misc:Depends}, vitastor-client (= ${binary:Version})
|
|
||||||
Description: Vitastor Proxmox Virtual Environment storage plugin
|
|
||||||
Vitastor storage plugin for Proxmox Virtual Environment.
|
|
||||||
|
|
|
@ -1 +0,0 @@
|
||||||
dep:fio=3.16-1
|
|
|
@ -1,4 +1,3 @@
|
||||||
VNPL-1.1.txt usr/share/doc/vitastor
|
VNPL-1.1.txt usr/share/doc/vitastor
|
||||||
GPL-2.0.txt usr/share/doc/vitastor
|
GPL-2.0.txt usr/share/doc/vitastor
|
||||||
README.md usr/share/doc/vitastor
|
mon usr/lib/vitastor
|
||||||
README-ru.md usr/share/doc/vitastor
|
|
||||||
|
|
|
@ -1,40 +0,0 @@
|
||||||
# Build patched libvirt for Debian Buster or Bullseye/Sid inside a container
|
|
||||||
# cd ..; podman build --build-arg REL=bullseye -v `pwd`/packages:/root/packages -f debian/libvirt.Dockerfile .
|
|
||||||
|
|
||||||
ARG REL=
|
|
||||||
FROM debian:$REL
|
|
||||||
ARG REL=
|
|
||||||
|
|
||||||
WORKDIR /root
|
|
||||||
|
|
||||||
RUN if [ "$REL" = "buster" -o "$REL" = "bullseye" ]; then \
|
|
||||||
echo "deb http://deb.debian.org/debian $REL-backports main" >> /etc/apt/sources.list; \
|
|
||||||
echo >> /etc/apt/preferences; \
|
|
||||||
echo 'Package: *' >> /etc/apt/preferences; \
|
|
||||||
echo "Pin: release a=$REL-backports" >> /etc/apt/preferences; \
|
|
||||||
echo 'Pin-Priority: 500' >> /etc/apt/preferences; \
|
|
||||||
fi; \
|
|
||||||
grep '^deb ' /etc/apt/sources.list | perl -pe 's/^deb/deb-src/' >> /etc/apt/sources.list; \
|
|
||||||
echo 'APT::Install-Recommends false;' >> /etc/apt/apt.conf; \
|
|
||||||
echo 'APT::Install-Suggests false;' >> /etc/apt/apt.conf
|
|
||||||
|
|
||||||
RUN apt-get update; apt-get -y install devscripts
|
|
||||||
RUN apt-get -y build-dep libvirt0
|
|
||||||
RUN apt-get -y install libglusterfs-dev
|
|
||||||
RUN apt-get --download-only source libvirt
|
|
||||||
|
|
||||||
ADD patches/libvirt-5.0-vitastor.diff patches/libvirt-7.0-vitastor.diff patches/libvirt-7.5-vitastor.diff patches/libvirt-7.6-vitastor.diff /root
|
|
||||||
RUN set -e; \
|
|
||||||
mkdir -p /root/packages/libvirt-$REL; \
|
|
||||||
rm -rf /root/packages/libvirt-$REL/*; \
|
|
||||||
cd /root/packages/libvirt-$REL; \
|
|
||||||
dpkg-source -x /root/libvirt*.dsc; \
|
|
||||||
D=$(ls -d libvirt-*/); \
|
|
||||||
V=$(ls -d libvirt-*/ | perl -pe 's/libvirt-(\d+\.\d+).*/$1/'); \
|
|
||||||
cp /root/libvirt-$V-vitastor.diff $D/debian/patches; \
|
|
||||||
echo libvirt-$V-vitastor.diff >> $D/debian/patches/series; \
|
|
||||||
cd $D; \
|
|
||||||
V=$(head -n1 debian/changelog | perl -pe 's/^.*\((.*?)(~bpo[\d\+]*)?(\+deb[u\d]+)?\).*$/$1/')+vitastor2; \
|
|
||||||
DEBEMAIL="Vitaliy Filippov <vitalif@yourcmc.ru>" dch -D $REL -v $V 'Add Vitastor support'; \
|
|
||||||
DEB_BUILD_OPTIONS=nocheck dpkg-buildpackage --jobs=auto -sa; \
|
|
||||||
rm -rf /root/packages/libvirt-$REL/$D
|
|
|
@ -1,17 +1,15 @@
|
||||||
# Build patched QEMU for Debian Buster or Bullseye/Sid inside a container
|
# Build patched QEMU for Debian Buster or Bullseye/Sid inside a container
|
||||||
# cd ..; podman build --build-arg REL=bullseye -v `pwd`/packages:/root/packages -f debian/patched-qemu.Dockerfile .
|
# cd ..; podman build --build-arg REL=bullseye -v `pwd`/packages:/root/packages -f debian/patched-qemu.Dockerfile .
|
||||||
|
|
||||||
ARG REL=
|
|
||||||
FROM debian:$REL
|
FROM debian:$REL
|
||||||
ARG REL=
|
|
||||||
|
|
||||||
WORKDIR /root
|
WORKDIR /root
|
||||||
|
|
||||||
RUN if [ "$REL" = "buster" -o "$REL" = "bullseye" ]; then \
|
RUN if [ "$REL" = "buster" ]; then \
|
||||||
echo "deb http://deb.debian.org/debian $REL-backports main" >> /etc/apt/sources.list; \
|
echo 'deb http://deb.debian.org/debian buster-backports main' >> /etc/apt/sources.list; \
|
||||||
echo >> /etc/apt/preferences; \
|
echo >> /etc/apt/preferences; \
|
||||||
echo 'Package: *' >> /etc/apt/preferences; \
|
echo 'Package: *' >> /etc/apt/preferences; \
|
||||||
echo "Pin: release a=$REL-backports" >> /etc/apt/preferences; \
|
echo 'Pin: release a=buster-backports' >> /etc/apt/preferences; \
|
||||||
echo 'Pin-Priority: 500' >> /etc/apt/preferences; \
|
echo 'Pin-Priority: 500' >> /etc/apt/preferences; \
|
||||||
fi; \
|
fi; \
|
||||||
grep '^deb ' /etc/apt/sources.list | perl -pe 's/^deb/deb-src/' >> /etc/apt/sources.list; \
|
grep '^deb ' /etc/apt/sources.list | perl -pe 's/^deb/deb-src/' >> /etc/apt/sources.list; \
|
||||||
|
@ -21,41 +19,26 @@ RUN if [ "$REL" = "buster" -o "$REL" = "bullseye" ]; then \
|
||||||
RUN apt-get update
|
RUN apt-get update
|
||||||
RUN apt-get -y install qemu fio liburing1 liburing-dev libgoogle-perftools-dev devscripts
|
RUN apt-get -y install qemu fio liburing1 liburing-dev libgoogle-perftools-dev devscripts
|
||||||
RUN apt-get -y build-dep qemu
|
RUN apt-get -y build-dep qemu
|
||||||
# To build a custom version
|
RUN apt-get -y build-dep fio
|
||||||
#RUN cp /root/packages/qemu-orig/* /root
|
|
||||||
RUN apt-get --download-only source qemu
|
RUN apt-get --download-only source qemu
|
||||||
|
RUN apt-get --download-only source fio
|
||||||
|
|
||||||
ADD patches/qemu-5.0-vitastor.patch patches/qemu-5.1-vitastor.patch patches/qemu-6.1-vitastor.patch src/qemu_driver.c /root/vitastor/patches/
|
ADD qemu-5.0-vitastor.patch qemu-5.1-vitastor.patch /root/vitastor/
|
||||||
RUN set -e; \
|
RUN set -e; \
|
||||||
apt-get install -y wget; \
|
|
||||||
wget -q -O /etc/apt/trusted.gpg.d/vitastor.gpg https://vitastor.io/debian/pubkey.gpg; \
|
|
||||||
(echo deb http://vitastor.io/debian $REL main > /etc/apt/sources.list.d/vitastor.list); \
|
|
||||||
(echo "APT::Install-Recommends false;" > /etc/apt/apt.conf) && \
|
|
||||||
apt-get update; \
|
|
||||||
apt-get install -y vitastor-client vitastor-client-dev quilt; \
|
|
||||||
mkdir -p /root/packages/qemu-$REL; \
|
mkdir -p /root/packages/qemu-$REL; \
|
||||||
rm -rf /root/packages/qemu-$REL/*; \
|
rm -rf /root/packages/qemu-$REL/*; \
|
||||||
cd /root/packages/qemu-$REL; \
|
cd /root/packages/qemu-$REL; \
|
||||||
dpkg-source -x /root/qemu*.dsc; \
|
dpkg-source -x /root/qemu*.dsc; \
|
||||||
if ls -d /root/packages/qemu-$REL/qemu-5.0*; then \
|
if [ -d /root/packages/qemu-$REL/qemu-5.0 ]; then \
|
||||||
D=$(ls -d /root/packages/qemu-$REL/qemu-5.0*); \
|
cp /root/vitastor/qemu-5.0-vitastor.patch /root/packages/qemu-$REL/qemu-5.0/debian/patches; \
|
||||||
cp /root/vitastor/patches/qemu-5.0-vitastor.patch $D/debian/patches; \
|
echo qemu-5.0-vitastor.patch >> /root/packages/qemu-$REL/qemu-5.0/debian/patches/series; \
|
||||||
echo qemu-5.0-vitastor.patch >> $D/debian/patches/series; \
|
|
||||||
elif ls /root/packages/qemu-$REL/qemu-6.1*; then \
|
|
||||||
D=$(ls -d /root/packages/qemu-$REL/qemu-6.1*); \
|
|
||||||
cp /root/vitastor/patches/qemu-6.1-vitastor.patch $D/debian/patches; \
|
|
||||||
echo qemu-6.1-vitastor.patch >> $D/debian/patches/series; \
|
|
||||||
else \
|
else \
|
||||||
cp /root/vitastor/patches/qemu-5.1-vitastor.patch /root/packages/qemu-$REL/qemu-*/debian/patches; \
|
cp /root/vitastor/qemu-5.1-vitastor.patch /root/packages/qemu-$REL/qemu-*/debian/patches; \
|
||||||
P=`ls -d /root/packages/qemu-$REL/qemu-*/debian/patches`; \
|
P=`ls -d /root/packages/qemu-$REL/qemu-*/debian/patches`; \
|
||||||
echo qemu-5.1-vitastor.patch >> $P/series; \
|
echo qemu-5.1-vitastor.patch >> $P/series; \
|
||||||
fi; \
|
fi; \
|
||||||
cd /root/packages/qemu-$REL/qemu-*/; \
|
cd /root/packages/qemu-$REL/qemu-*/; \
|
||||||
quilt push -a; \
|
|
||||||
quilt add block/vitastor.c; \
|
|
||||||
cp /root/vitastor/patches/qemu_driver.c block/vitastor.c; \
|
|
||||||
quilt refresh; \
|
|
||||||
V=$(head -n1 debian/changelog | perl -pe 's/^.*\((.*?)(~bpo[\d\+]*)?\).*$/$1/')+vitastor1; \
|
V=$(head -n1 debian/changelog | perl -pe 's/^.*\((.*?)(~bpo[\d\+]*)?\).*$/$1/')+vitastor1; \
|
||||||
DEBEMAIL="Vitaliy Filippov <vitalif@yourcmc.ru>" dch -D $REL -v $V 'Plug Vitastor block driver'; \
|
DEBFULLNAME="Vitaliy Filippov <vitalif@yourcmc.ru>" dch -D $REL -v $V 'Plug Vitastor block driver'; \
|
||||||
DEB_BUILD_OPTIONS=nocheck dpkg-buildpackage --jobs=auto -sa; \
|
DEB_BUILD_OPTIONS=nocheck dpkg-buildpackage --jobs=auto -sa; \
|
||||||
rm -rf /root/packages/qemu-$REL/qemu-*/
|
rm -rf /root/packages/qemu-$REL/qemu-*/
|
||||||
|
|
|
@ -1 +0,0 @@
|
||||||
patches/PVE_VitastorPlugin.pm usr/share/perl5/PVE/Storage/Custom/VitastorPlugin.pm
|
|
|
@ -1,19 +0,0 @@
|
||||||
/* Removed in Linux 5.14 */
|
|
||||||
|
|
||||||
/* SPDX-License-Identifier: GPL-2.0 WITH Linux-syscall-note */
|
|
||||||
#ifndef __LINUX_RAW_H
|
|
||||||
#define __LINUX_RAW_H
|
|
||||||
|
|
||||||
#include <linux/types.h>
|
|
||||||
|
|
||||||
#define RAW_SETBIND _IO( 0xac, 0 )
|
|
||||||
#define RAW_GETBIND _IO( 0xac, 1 )
|
|
||||||
|
|
||||||
struct raw_config_request
|
|
||||||
{
|
|
||||||
int raw_minor;
|
|
||||||
__u64 block_major;
|
|
||||||
__u64 block_minor;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif /* __LINUX_RAW_H */
|
|
|
@ -5,6 +5,5 @@ export DH_VERBOSE = 1
|
||||||
dh $@
|
dh $@
|
||||||
|
|
||||||
override_dh_installdeb:
|
override_dh_installdeb:
|
||||||
cat debian/fio_version >> debian/vitastor-fio.substvars
|
cat debian/substvars >> debian/vitastor.substvars
|
||||||
[ -f debian/qemu_version ] && (cat debian/qemu_version >> debian/vitastor-qemu.substvars) || true
|
|
||||||
dh_installdeb
|
dh_installdeb
|
||||||
|
|
|
@ -0,0 +1,2 @@
|
||||||
|
dep:fio=3.16-1
|
||||||
|
dep:qemu=1:5.1+dfsg-4+vitastor1
|
|
@ -1,2 +0,0 @@
|
||||||
usr/include
|
|
||||||
usr/lib/*/pkgconfig
|
|
|
@ -1,6 +0,0 @@
|
||||||
usr/bin/vita
|
|
||||||
usr/bin/vitastor-cli
|
|
||||||
usr/bin/vitastor-rm
|
|
||||||
usr/bin/vitastor-nbd
|
|
||||||
usr/lib/*/libvitastor*.so*
|
|
||||||
mon/make-osd.sh /usr/lib/vitastor
|
|
|
@ -1 +0,0 @@
|
||||||
usr/lib/*/libfio*.so*
|
|
|
@ -1 +0,0 @@
|
||||||
mon usr/lib/vitastor
|
|
|
@ -1,2 +0,0 @@
|
||||||
usr/bin/vitastor-osd
|
|
||||||
usr/bin/vitastor-dump-journal
|
|
|
@ -1,17 +1,15 @@
|
||||||
# Build Vitastor packages for Debian Buster or Bullseye/Sid inside a container
|
# Build Vitastor packages for Debian Buster or Bullseye/Sid inside a container
|
||||||
# cd ..; podman build --build-arg REL=bullseye -v `pwd`/packages:/root/packages -f debian/vitastor.Dockerfile .
|
# cd ..; podman build --build-arg REL=bullseye -v `pwd`/packages:/root/packages -f debian/vitastor.Dockerfile .
|
||||||
|
|
||||||
ARG REL=
|
|
||||||
FROM debian:$REL
|
FROM debian:$REL
|
||||||
ARG REL=
|
|
||||||
|
|
||||||
WORKDIR /root
|
WORKDIR /root
|
||||||
|
|
||||||
RUN if [ "$REL" = "buster" -o "$REL" = "bullseye" ]; then \
|
RUN if [ "$REL" = "buster" ]; then \
|
||||||
echo "deb http://deb.debian.org/debian $REL-backports main" >> /etc/apt/sources.list; \
|
echo 'deb http://deb.debian.org/debian buster-backports main' >> /etc/apt/sources.list; \
|
||||||
echo >> /etc/apt/preferences; \
|
echo >> /etc/apt/preferences; \
|
||||||
echo 'Package: *' >> /etc/apt/preferences; \
|
echo 'Package: *' >> /etc/apt/preferences; \
|
||||||
echo "Pin: release a=$REL-backports" >> /etc/apt/preferences; \
|
echo 'Pin: release a=buster-backports' >> /etc/apt/preferences; \
|
||||||
echo 'Pin-Priority: 500' >> /etc/apt/preferences; \
|
echo 'Pin-Priority: 500' >> /etc/apt/preferences; \
|
||||||
fi; \
|
fi; \
|
||||||
grep '^deb ' /etc/apt/sources.list | perl -pe 's/^deb/deb-src/' >> /etc/apt/sources.list; \
|
grep '^deb ' /etc/apt/sources.list | perl -pe 's/^deb/deb-src/' >> /etc/apt/sources.list; \
|
||||||
|
@ -19,10 +17,12 @@ RUN if [ "$REL" = "buster" -o "$REL" = "bullseye" ]; then \
|
||||||
echo 'APT::Install-Suggests false;' >> /etc/apt/apt.conf
|
echo 'APT::Install-Suggests false;' >> /etc/apt/apt.conf
|
||||||
|
|
||||||
RUN apt-get update
|
RUN apt-get update
|
||||||
RUN apt-get -y install fio liburing1 liburing-dev libgoogle-perftools-dev devscripts
|
RUN apt-get -y install qemu fio liburing1 liburing-dev libgoogle-perftools-dev devscripts
|
||||||
|
RUN apt-get -y build-dep qemu
|
||||||
RUN apt-get -y build-dep fio
|
RUN apt-get -y build-dep fio
|
||||||
|
RUN apt-get --download-only source qemu
|
||||||
RUN apt-get --download-only source fio
|
RUN apt-get --download-only source fio
|
||||||
RUN apt-get update && apt-get -y install libjerasure-dev cmake libibverbs-dev
|
RUN apt-get -y install libjerasure-dev cmake
|
||||||
|
|
||||||
ADD . /root/vitastor
|
ADD . /root/vitastor
|
||||||
RUN set -e -x; \
|
RUN set -e -x; \
|
||||||
|
@ -30,25 +30,37 @@ RUN set -e -x; \
|
||||||
cd /root/fio-build/; \
|
cd /root/fio-build/; \
|
||||||
rm -rf /root/fio-build/*; \
|
rm -rf /root/fio-build/*; \
|
||||||
dpkg-source -x /root/fio*.dsc; \
|
dpkg-source -x /root/fio*.dsc; \
|
||||||
|
cd /root/packages/qemu-$REL/; \
|
||||||
|
rm -rf qemu*/; \
|
||||||
|
dpkg-source -x qemu*.dsc; \
|
||||||
|
cd /root/packages/qemu-$REL/qemu*/; \
|
||||||
|
debian/rules b/configure-stamp; \
|
||||||
|
cd b/qemu; \
|
||||||
|
make -j8 qapi/qapi-builtin-types.h; \
|
||||||
mkdir -p /root/packages/vitastor-$REL; \
|
mkdir -p /root/packages/vitastor-$REL; \
|
||||||
rm -rf /root/packages/vitastor-$REL/*; \
|
rm -rf /root/packages/vitastor-$REL/*; \
|
||||||
cd /root/packages/vitastor-$REL; \
|
cd /root/packages/vitastor-$REL; \
|
||||||
cp -r /root/vitastor vitastor-0.6.10; \
|
cp -r /root/vitastor vitastor-0.6.2; \
|
||||||
cd vitastor-0.6.10; \
|
ln -s /root/packages/qemu-$REL/qemu-*/ vitastor-0.6.2/qemu; \
|
||||||
ln -s /root/fio-build/fio-*/ ./fio; \
|
ln -s /root/fio-build/fio-*/ vitastor-0.6.2/fio; \
|
||||||
|
cd vitastor-0.6.2; \
|
||||||
FIO=$(head -n1 fio/debian/changelog | perl -pe 's/^.*\((.*?)\).*$/$1/'); \
|
FIO=$(head -n1 fio/debian/changelog | perl -pe 's/^.*\((.*?)\).*$/$1/'); \
|
||||||
ls /usr/include/linux/raw.h || cp ./debian/raw.h /usr/include/linux/raw.h; \
|
QEMU=$(head -n1 qemu/debian/changelog | perl -pe 's/^.*\((.*?)\).*$/$1/'); \
|
||||||
|
sh copy-qemu-includes.sh; \
|
||||||
sh copy-fio-includes.sh; \
|
sh copy-fio-includes.sh; \
|
||||||
rm fio; \
|
rm qemu fio; \
|
||||||
mkdir -p a b debian/patches; \
|
mkdir -p a b debian/patches; \
|
||||||
|
mv qemu-copy b/qemu; \
|
||||||
mv fio-copy b/fio; \
|
mv fio-copy b/fio; \
|
||||||
diff -NaurpbB a b > debian/patches/fio-headers.patch || true; \
|
diff -NaurpbB a b > debian/patches/qemu-fio-headers.patch || true; \
|
||||||
echo fio-headers.patch >> debian/patches/series; \
|
echo qemu-fio-headers.patch >> debian/patches/series; \
|
||||||
rm -rf a b; \
|
rm -rf a b; \
|
||||||
echo "dep:fio=$FIO" > debian/fio_version; \
|
rm -rf /root/packages/qemu-$REL/qemu*/; \
|
||||||
|
echo "dep:fio=$FIO" > debian/substvars; \
|
||||||
|
echo "dep:qemu=$QEMU" >> debian/substvars; \
|
||||||
cd /root/packages/vitastor-$REL; \
|
cd /root/packages/vitastor-$REL; \
|
||||||
tar --sort=name --mtime='2020-01-01' --owner=0 --group=0 --exclude=debian -cJf vitastor_0.6.10.orig.tar.xz vitastor-0.6.10; \
|
tar --sort=name --mtime='2020-01-01' --owner=0 --group=0 --exclude=debian -cJf vitastor_0.6.2.orig.tar.xz vitastor-0.6.2; \
|
||||||
cd vitastor-0.6.10; \
|
cd vitastor-0.6.2; \
|
||||||
V=$(head -n1 debian/changelog | perl -pe 's/^.*\((.*?)\).*$/$1/'); \
|
V=$(head -n1 debian/changelog | perl -pe 's/^.*\((.*?)\).*$/$1/'); \
|
||||||
DEBFULLNAME="Vitaliy Filippov <vitalif@yourcmc.ru>" dch -D $REL -v "$V""$REL" "Rebuild for $REL"; \
|
DEBFULLNAME="Vitaliy Filippov <vitalif@yourcmc.ru>" dch -D $REL -v "$V""$REL" "Rebuild for $REL"; \
|
||||||
DEB_BUILD_OPTIONS=nocheck dpkg-buildpackage --jobs=auto -sa; \
|
DEB_BUILD_OPTIONS=nocheck dpkg-buildpackage --jobs=auto -sa; \
|
||||||
|
|
|
@ -1,9 +0,0 @@
|
||||||
# Build Docker image with Vitastor packages
|
|
||||||
|
|
||||||
FROM debian:bullseye
|
|
||||||
|
|
||||||
ADD vitastor.list /etc/apt/sources.list.d
|
|
||||||
ADD vitastor.gpg /etc/apt/trusted.gpg.d
|
|
||||||
ADD vitastor.pref /etc/apt/preferences.d
|
|
||||||
ADD apt.conf /etc/apt/
|
|
||||||
RUN apt-get update && apt-get -y install vitastor qemu-system-x86 qemu-system-common && apt-get clean
|
|
|
@ -1 +0,0 @@
|
||||||
APT::Install-Recommends false;
|
|
Binary file not shown.
|
@ -1 +0,0 @@
|
||||||
deb http://vitastor.io/debian bullseye main
|
|
|
@ -1,3 +0,0 @@
|
||||||
Package: *
|
|
||||||
Pin: origin "vitastor.io"
|
|
||||||
Pin-Priority: 1000
|
|
2
json11
2
json11
|
@ -1 +1 @@
|
||||||
Subproject commit 55363fc2653b8802637a3d2e73a06839a72c585d
|
Subproject commit 97f06cb20c1e136fd37d58fb40f57dd8f8a3a4a7
|
|
@ -50,7 +50,7 @@ async function lp_solve(text)
|
||||||
return { score, vars };
|
return { score, vars };
|
||||||
}
|
}
|
||||||
|
|
||||||
async function optimize_initial({ osd_tree, pg_count, pg_size = 3, pg_minsize = 2, max_combinations = 10000, parity_space = 1, round_robin = false })
|
async function optimize_initial({ osd_tree, pg_count, pg_size = 3, pg_minsize = 2, max_combinations = 10000, parity_space = 1 })
|
||||||
{
|
{
|
||||||
if (!pg_count || !osd_tree)
|
if (!pg_count || !osd_tree)
|
||||||
{
|
{
|
||||||
|
@ -92,7 +92,7 @@ async function optimize_initial({ osd_tree, pg_count, pg_size = 3, pg_minsize =
|
||||||
console.log(lp);
|
console.log(lp);
|
||||||
throw new Error('Problem is infeasible or unbounded - is it a bug?');
|
throw new Error('Problem is infeasible or unbounded - is it a bug?');
|
||||||
}
|
}
|
||||||
const int_pgs = make_int_pgs(lp_result.vars, pg_count, round_robin);
|
const int_pgs = make_int_pgs(lp_result.vars, pg_count);
|
||||||
const eff = pg_list_space_efficiency(int_pgs, all_weights, pg_minsize, parity_space);
|
const eff = pg_list_space_efficiency(int_pgs, all_weights, pg_minsize, parity_space);
|
||||||
const res = {
|
const res = {
|
||||||
score: lp_result.score,
|
score: lp_result.score,
|
||||||
|
@ -115,7 +115,7 @@ function shuffle(array)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function make_int_pgs(weights, pg_count, round_robin)
|
function make_int_pgs(weights, pg_count)
|
||||||
{
|
{
|
||||||
const total_weight = Object.values(weights).reduce((a, c) => Number(a) + Number(c), 0);
|
const total_weight = Object.values(weights).reduce((a, c) => Number(a) + Number(c), 0);
|
||||||
let int_pgs = [];
|
let int_pgs = [];
|
||||||
|
@ -123,15 +123,10 @@ function make_int_pgs(weights, pg_count, round_robin)
|
||||||
let weight_left = total_weight;
|
let weight_left = total_weight;
|
||||||
for (const pg_name in weights)
|
for (const pg_name in weights)
|
||||||
{
|
{
|
||||||
let cur_pg = pg_name.substr(3).split('_');
|
|
||||||
let n = Math.round(weights[pg_name] / weight_left * pg_left);
|
let n = Math.round(weights[pg_name] / weight_left * pg_left);
|
||||||
for (let i = 0; i < n; i++)
|
for (let i = 0; i < n; i++)
|
||||||
{
|
{
|
||||||
int_pgs.push([ ...cur_pg ]);
|
int_pgs.push(pg_name.substr(3).split('_'));
|
||||||
if (round_robin)
|
|
||||||
{
|
|
||||||
cur_pg.push(cur_pg.shift());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
weight_left -= weights[pg_name];
|
weight_left -= weights[pg_name];
|
||||||
pg_left -= n;
|
pg_left -= n;
|
||||||
|
@ -249,7 +244,6 @@ async function optimize_change({ prev_pgs: prev_int_pgs, osd_tree, pg_size = 3,
|
||||||
{
|
{
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
// FIXME: use parity_chunks with parity_space instead of pg_minsize
|
|
||||||
const pg_effsize = Math.min(pg_minsize, Object.keys(osd_tree).length)
|
const pg_effsize = Math.min(pg_minsize, Object.keys(osd_tree).length)
|
||||||
+ Math.max(0, Math.min(pg_size, Object.keys(osd_tree).length) - pg_minsize) * parity_space;
|
+ Math.max(0, Math.min(pg_size, Object.keys(osd_tree).length) - pg_minsize) * parity_space;
|
||||||
const pg_count = prev_int_pgs.length;
|
const pg_count = prev_int_pgs.length;
|
||||||
|
@ -293,6 +287,11 @@ async function optimize_change({ prev_pgs: prev_int_pgs, osd_tree, pg_size = 3,
|
||||||
lp += 'max: '+all_pg_names.map(pg_name => (
|
lp += 'max: '+all_pg_names.map(pg_name => (
|
||||||
prev_weights[pg_name] ? `${pg_size+1}*add_${pg_name} - ${pg_size+1}*del_${pg_name}` : `${pg_size+1-move_weights[pg_name]}*${pg_name}`
|
prev_weights[pg_name] ? `${pg_size+1}*add_${pg_name} - ${pg_size+1}*del_${pg_name}` : `${pg_size+1-move_weights[pg_name]}*${pg_name}`
|
||||||
)).join(' + ')+';\n';
|
)).join(' + ')+';\n';
|
||||||
|
lp += all_pg_names
|
||||||
|
.map(pg_name => (prev_weights[pg_name] ? `add_${pg_name} - del_${pg_name}` : `${pg_name}`))
|
||||||
|
.join(' + ')+' = '+(pg_count
|
||||||
|
- Object.keys(prev_weights).reduce((a, old_pg_name) => (a + (all_pgs_hash[old_pg_name] ? prev_weights[old_pg_name] : 0)), 0)
|
||||||
|
)+';\n';
|
||||||
for (const osd in pg_per_osd)
|
for (const osd in pg_per_osd)
|
||||||
{
|
{
|
||||||
if (osd !== NO_OSD)
|
if (osd !== NO_OSD)
|
||||||
|
|
|
@ -4,25 +4,36 @@
|
||||||
# Copyright (c) Vitaliy Filippov, 2019+
|
# Copyright (c) Vitaliy Filippov, 2019+
|
||||||
# License: MIT
|
# License: MIT
|
||||||
|
|
||||||
# USAGE:
|
# USAGE: ./make-osd.sh /dev/disk/by-partuuid/xxx [ /dev/disk/by-partuuid/yyy]...
|
||||||
# 1) Put etcd_address and osd_network into /etc/vitastor/vitastor.conf. Example:
|
|
||||||
# {
|
IP_SUBSTR="10.200.1."
|
||||||
# "etcd_address":["http://10.200.1.10:2379/v3","http://10.200.1.11:2379/v3","http://10.200.1.12:2379/v3"],
|
ETCD_HOSTS="etcd0=http://10.200.1.10:2380,etcd1=http://10.200.1.11:2380,etcd2=http://10.200.1.12:2380"
|
||||||
# "osd_network":"10.200.1.0/24"
|
|
||||||
# }
|
|
||||||
# 2) Run ./make-osd.sh /dev/disk/by-partuuid/xxx [ /dev/disk/by-partuuid/yyy]...
|
|
||||||
|
|
||||||
set -e -x
|
set -e -x
|
||||||
|
|
||||||
|
IP=`ip -json a s | jq -r '.[].addr_info[] | select(.local | startswith("'$IP_SUBSTR'")) | .local'`
|
||||||
|
[ "$IP" != "" ] || exit 1
|
||||||
|
ETCD_MON=$(echo $ETCD_HOSTS | perl -pe 's/:2380/:2379/g; s/etcd\d*=//g;')
|
||||||
|
D=`dirname $0`
|
||||||
|
|
||||||
# Create OSDs on all passed devices
|
# Create OSDs on all passed devices
|
||||||
|
OSD_NUM=1
|
||||||
for DEV in $*; do
|
for DEV in $*; do
|
||||||
|
|
||||||
OSD_NUM=$(vitastor-cli alloc-osd)
|
# Ugly :) -> node.js rework pending
|
||||||
|
while true; do
|
||||||
|
ST=$(etcdctl --endpoints="$ETCD_MON" get --print-value-only /vitastor/osd/stats/$OSD_NUM)
|
||||||
|
if [ "$ST" = "" ]; then
|
||||||
|
break
|
||||||
|
fi
|
||||||
|
OSD_NUM=$((OSD_NUM+1))
|
||||||
|
done
|
||||||
|
etcdctl --endpoints="$ETCD_MON" put /vitastor/osd/stats/$OSD_NUM '{}'
|
||||||
|
|
||||||
echo Creating OSD $OSD_NUM on $DEV
|
echo Creating OSD $OSD_NUM on $DEV
|
||||||
|
|
||||||
OPT=$(vitastor-cli simple-offsets --format options $DEV | tr '\n' ' ')
|
OPT=`node $D/simple-offsets.js --device $DEV --format options | tr '\n' ' '`
|
||||||
META=$(vitastor-cli simple-offsets --format json $DEV | jq .data_offset)
|
META=`echo $OPT | grep -Po '(?<=data_offset )\d+'`
|
||||||
dd if=/dev/zero of=$DEV bs=1048576 count=$(((META+1048575)/1048576)) oflag=direct
|
dd if=/dev/zero of=$DEV bs=1048576 count=$(((META+1048575)/1048576)) oflag=direct
|
||||||
|
|
||||||
cat >/etc/systemd/system/vitastor-osd$OSD_NUM.service <<EOF
|
cat >/etc/systemd/system/vitastor-osd$OSD_NUM.service <<EOF
|
||||||
|
@ -37,6 +48,8 @@ LimitNOFILE=1048576
|
||||||
LimitNPROC=1048576
|
LimitNPROC=1048576
|
||||||
LimitMEMLOCK=infinity
|
LimitMEMLOCK=infinity
|
||||||
ExecStart=/usr/bin/vitastor-osd \\
|
ExecStart=/usr/bin/vitastor-osd \\
|
||||||
|
--etcd_address $IP:2379/v3 \\
|
||||||
|
--bind_address $IP \\
|
||||||
--osd_num $OSD_NUM \\
|
--osd_num $OSD_NUM \\
|
||||||
--disable_data_fsync 1 \\
|
--disable_data_fsync 1 \\
|
||||||
--immediate_commit all \\
|
--immediate_commit all \\
|
||||||
|
|
|
@ -9,18 +9,17 @@ const options = {};
|
||||||
|
|
||||||
for (let i = 2; i < process.argv.length; i++)
|
for (let i = 2; i < process.argv.length; i++)
|
||||||
{
|
{
|
||||||
if (process.argv[i] === '-h' || process.argv[i] === '--help')
|
if (process.argv[i].substr(0, 2) == '--')
|
||||||
{
|
|
||||||
console.error('USAGE: '+process.argv[0]+' '+process.argv[1]+' [--verbose 1]'+
|
|
||||||
' [--etcd_address "http://127.0.0.1:2379,..."] [--config_file /etc/vitastor/vitastor.conf]'+
|
|
||||||
' [--etcd_prefix "/vitastor"] [--etcd_start_timeout 5]');
|
|
||||||
process.exit();
|
|
||||||
}
|
|
||||||
else if (process.argv[i].substr(0, 2) == '--')
|
|
||||||
{
|
{
|
||||||
options[process.argv[i].substr(2)] = process.argv[i+1];
|
options[process.argv[i].substr(2)] = process.argv[i+1];
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
new Mon(options).start().catch(e => { console.error(e); process.exit(1); });
|
if (!options.etcd_url)
|
||||||
|
{
|
||||||
|
console.error('USAGE: '+process.argv[0]+' '+process.argv[1]+' --etcd_url "http://127.0.0.1:2379,..." --etcd_prefix "/vitastor" --etcd_start_timeout 5 [--verbose 1]');
|
||||||
|
process.exit();
|
||||||
|
}
|
||||||
|
|
||||||
|
new Mon(options).start().catch(e => { console.error(e); process.exit(); });
|
||||||
|
|
403
mon/mon.js
403
mon/mon.js
|
@ -1,7 +1,6 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
// Copyright (c) Vitaliy Filippov, 2019+
|
||||||
// License: VNPL-1.1 (see README.md for details)
|
// License: VNPL-1.1 (see README.md for details)
|
||||||
|
|
||||||
const fs = require('fs');
|
|
||||||
const http = require('http');
|
const http = require('http');
|
||||||
const crypto = require('crypto');
|
const crypto = require('crypto');
|
||||||
const os = require('os');
|
const os = require('os');
|
||||||
|
@ -34,24 +33,14 @@ const etcd_allow = new RegExp('^'+[
|
||||||
'pg/state/[1-9]\\d*/[1-9]\\d*',
|
'pg/state/[1-9]\\d*/[1-9]\\d*',
|
||||||
'pg/stats/[1-9]\\d*/[1-9]\\d*',
|
'pg/stats/[1-9]\\d*/[1-9]\\d*',
|
||||||
'pg/history/[1-9]\\d*/[1-9]\\d*',
|
'pg/history/[1-9]\\d*/[1-9]\\d*',
|
||||||
'pool/stats/[1-9]\\d*',
|
|
||||||
'history/last_clean_pgs',
|
'history/last_clean_pgs',
|
||||||
'inode/stats/[1-9]\\d*/[1-9]\\d*',
|
'inode/stats/[1-9]\\d*/[1-9]\\d*',
|
||||||
'pool/stats/[1-9]\\d*',
|
|
||||||
'stats',
|
'stats',
|
||||||
'index/image/.*',
|
|
||||||
'index/maxid/[1-9]\\d*',
|
|
||||||
].join('$|^')+'$');
|
].join('$|^')+'$');
|
||||||
|
|
||||||
const etcd_tree = {
|
const etcd_tree = {
|
||||||
config: {
|
config: {
|
||||||
/* global: {
|
/* global: {
|
||||||
// WARNING: NOT ALL OF THESE ARE ACTUALLY CONFIGURABLE HERE
|
|
||||||
// THIS IS JUST A POOR MAN'S CONFIG DOCUMENTATION
|
|
||||||
// etcd connection
|
|
||||||
config_path: "/etc/vitastor/vitastor.conf",
|
|
||||||
etcd_address: "10.0.115.10:2379/v3",
|
|
||||||
etcd_prefix: "/vitastor",
|
|
||||||
// mon
|
// mon
|
||||||
etcd_mon_ttl: 30, // min: 10
|
etcd_mon_ttl: 30, // min: 10
|
||||||
etcd_mon_timeout: 1000, // ms. min: 0
|
etcd_mon_timeout: 1000, // ms. min: 0
|
||||||
|
@ -61,17 +50,7 @@ const etcd_tree = {
|
||||||
osd_out_time: 600, // seconds. min: 0
|
osd_out_time: 600, // seconds. min: 0
|
||||||
placement_levels: { datacenter: 1, rack: 2, host: 3, osd: 4, ... },
|
placement_levels: { datacenter: 1, rack: 2, host: 3, osd: 4, ... },
|
||||||
// client and osd
|
// client and osd
|
||||||
tcp_header_buffer_size: 65536,
|
|
||||||
use_sync_send_recv: false,
|
use_sync_send_recv: false,
|
||||||
use_rdma: true,
|
|
||||||
rdma_device: null, // for example, "rocep5s0f0"
|
|
||||||
rdma_port_num: 1,
|
|
||||||
rdma_gid_index: 0,
|
|
||||||
rdma_mtu: 4096,
|
|
||||||
rdma_max_sge: 128,
|
|
||||||
rdma_max_send: 32,
|
|
||||||
rdma_max_recv: 8,
|
|
||||||
rdma_max_msg: 1048576,
|
|
||||||
log_level: 0,
|
log_level: 0,
|
||||||
block_size: 131072,
|
block_size: 131072,
|
||||||
disk_alignment: 4096,
|
disk_alignment: 4096,
|
||||||
|
@ -84,13 +63,11 @@ const etcd_tree = {
|
||||||
osd_ping_timeout: 5, // seconds. min: 1
|
osd_ping_timeout: 5, // seconds. min: 1
|
||||||
up_wait_retry_interval: 500, // ms. min: 50
|
up_wait_retry_interval: 500, // ms. min: 50
|
||||||
// osd
|
// osd
|
||||||
etcd_report_interval: 5,
|
etcd_report_interval: 30, // min: 10
|
||||||
run_primary: true,
|
run_primary: true,
|
||||||
osd_network: null, // "192.168.7.0/24" or an array of masks
|
|
||||||
bind_address: "0.0.0.0",
|
bind_address: "0.0.0.0",
|
||||||
bind_port: 0,
|
bind_port: 0,
|
||||||
autosync_interval: 5,
|
autosync_interval: 5,
|
||||||
autosync_writes: 128,
|
|
||||||
client_queue_depth: 128, // unused
|
client_queue_depth: 128, // unused
|
||||||
recovery_queue_depth: 4,
|
recovery_queue_depth: 4,
|
||||||
recovery_sync_batch: 16,
|
recovery_sync_batch: 16,
|
||||||
|
@ -234,7 +211,7 @@ const etcd_tree = {
|
||||||
/* <pool_id>: {
|
/* <pool_id>: {
|
||||||
<pg_id>: {
|
<pg_id>: {
|
||||||
primary: osd_num_t,
|
primary: osd_num_t,
|
||||||
state: ("starting"|"peering"|"peered"|"incomplete"|"active"|"repeering"|"stopping"|"offline"|
|
state: ("starting"|"peering"|"incomplete"|"active"|"repeering"|"stopping"|"offline"|
|
||||||
"degraded"|"has_incomplete"|"has_degraded"|"has_misplaced"|"has_unclean"|
|
"degraded"|"has_incomplete"|"has_degraded"|"has_misplaced"|"has_unclean"|
|
||||||
"has_invalid"|"left_on_dead")[],
|
"has_invalid"|"left_on_dead")[],
|
||||||
}
|
}
|
||||||
|
@ -264,36 +241,24 @@ const etcd_tree = {
|
||||||
},
|
},
|
||||||
inode: {
|
inode: {
|
||||||
stats: {
|
stats: {
|
||||||
/* <pool_id>: {
|
/* <inode_t>: {
|
||||||
<inode_t>: {
|
|
||||||
raw_used: uint64_t, // raw used bytes on OSDs
|
raw_used: uint64_t, // raw used bytes on OSDs
|
||||||
read: { count: uint64_t, usec: uint64_t, bytes: uint64_t, bps: uint64_t, iops: uint64_t, lat: uint64_t },
|
read: { count: uint64_t, usec: uint64_t, bytes: uint64_t },
|
||||||
write: { count: uint64_t, usec: uint64_t, bytes: uint64_t, bps: uint64_t, iops: uint64_t, lat: uint64_t },
|
write: { count: uint64_t, usec: uint64_t, bytes: uint64_t },
|
||||||
delete: { count: uint64_t, usec: uint64_t, bytes: uint64_t, bps: uint64_t, iops: uint64_t, lat: uint64_t },
|
delete: { count: uint64_t, usec: uint64_t, bytes: uint64_t },
|
||||||
},
|
|
||||||
}, */
|
}, */
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
pool: {
|
|
||||||
stats: {
|
|
||||||
/* <pool_id>: {
|
|
||||||
used_raw_tb: float, // used raw space in the pool
|
|
||||||
total_raw_tb: float, // maximum amount of space in the pool
|
|
||||||
raw_to_usable: float, // raw to usable ratio
|
|
||||||
space_efficiency: float, // 0..1
|
|
||||||
} */
|
|
||||||
},
|
|
||||||
},
|
|
||||||
stats: {
|
stats: {
|
||||||
/* op_stats: {
|
/* op_stats: {
|
||||||
<string>: { count: uint64_t, usec: uint64_t, bytes: uint64_t, bps: uint64_t, iops: uint64_t, lat: uint64_t },
|
<string>: { count: uint64_t, usec: uint64_t, bytes: uint64_t },
|
||||||
},
|
},
|
||||||
subop_stats: {
|
subop_stats: {
|
||||||
<string>: { count: uint64_t, usec: uint64_t, iops: uint64_t, lat: uint64_t },
|
<string>: { count: uint64_t, usec: uint64_t },
|
||||||
},
|
},
|
||||||
recovery_stats: {
|
recovery_stats: {
|
||||||
degraded: { count: uint64_t, bytes: uint64_t, bps: uint64_t, iops: uint64_t },
|
degraded: { count: uint64_t, bytes: uint64_t },
|
||||||
misplaced: { count: uint64_t, bytes: uint64_t, bps: uint64_t, iops: uint64_t },
|
misplaced: { count: uint64_t, bytes: uint64_t },
|
||||||
},
|
},
|
||||||
object_counts: {
|
object_counts: {
|
||||||
object: uint64_t,
|
object: uint64_t,
|
||||||
|
@ -306,17 +271,6 @@ const etcd_tree = {
|
||||||
history: {
|
history: {
|
||||||
last_clean_pgs: {},
|
last_clean_pgs: {},
|
||||||
},
|
},
|
||||||
index: {
|
|
||||||
image: {
|
|
||||||
/* <name>: {
|
|
||||||
id: uint64_t,
|
|
||||||
pool_id: uint64_t,
|
|
||||||
}, */
|
|
||||||
},
|
|
||||||
maxid: {
|
|
||||||
/* <pool_id>: uint64_t, */
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// FIXME Split into several files
|
// FIXME Split into several files
|
||||||
|
@ -324,53 +278,22 @@ class Mon
|
||||||
{
|
{
|
||||||
constructor(config)
|
constructor(config)
|
||||||
{
|
{
|
||||||
this.die = (e) => this._die(e);
|
// FIXME: Maybe prefer local etcd
|
||||||
if (fs.existsSync(config.config_path||'/etc/vitastor/vitastor.conf'))
|
this.etcd_urls = [];
|
||||||
|
for (let url of config.etcd_url.split(/,/))
|
||||||
{
|
{
|
||||||
config = {
|
let scheme = 'http';
|
||||||
...JSON.parse(fs.readFileSync(config.config_path||'/etc/vitastor/vitastor.conf', { encoding: 'utf-8' })),
|
url = url.trim().replace(/^(https?):\/\//, (m, m1) => { scheme = m1; return ''; });
|
||||||
...config,
|
if (!/\/[^\/]/.exec(url))
|
||||||
};
|
url += '/v3';
|
||||||
|
this.etcd_urls.push(scheme+'://'+url);
|
||||||
}
|
}
|
||||||
this.parse_etcd_addresses(config.etcd_address||config.etcd_url);
|
|
||||||
this.verbose = config.verbose || 0;
|
this.verbose = config.verbose || 0;
|
||||||
this.initConfig = config;
|
|
||||||
this.config = {};
|
this.config = {};
|
||||||
this.etcd_prefix = config.etcd_prefix || '/vitastor';
|
this.etcd_prefix = config.etcd_prefix || '/vitastor';
|
||||||
this.etcd_prefix = this.etcd_prefix.replace(/\/\/+/g, '/').replace(/^\/?(.*[^\/])\/?$/, '/$1');
|
this.etcd_prefix = this.etcd_prefix.replace(/\/\/+/g, '/').replace(/^\/?(.*[^\/])\/?$/, '/$1');
|
||||||
this.etcd_start_timeout = (config.etcd_start_timeout || 5) * 1000;
|
this.etcd_start_timeout = (config.etcd_start_timeout || 5) * 1000;
|
||||||
this.state = JSON.parse(JSON.stringify(this.constructor.etcd_tree));
|
this.state = JSON.parse(JSON.stringify(this.constructor.etcd_tree));
|
||||||
this.signals_set = false;
|
|
||||||
this.on_stop_cb = () => this.on_stop(0).catch(console.error);
|
|
||||||
}
|
|
||||||
|
|
||||||
parse_etcd_addresses(addrs)
|
|
||||||
{
|
|
||||||
const is_local_ip = this.local_ips(true).reduce((a, c) => { a[c] = true; return a; }, {});
|
|
||||||
this.etcd_local = [];
|
|
||||||
this.etcd_urls = [];
|
|
||||||
this.selected_etcd_url = null;
|
|
||||||
this.etcd_urls_to_try = [];
|
|
||||||
if (!(addrs instanceof Array))
|
|
||||||
addrs = addrs ? (''+(addrs||'')).split(/,/) : [];
|
|
||||||
if (!addrs.length)
|
|
||||||
{
|
|
||||||
console.error('Vitastor etcd address(es) not specified. Please set on the command line or in the config file');
|
|
||||||
process.exit(1);
|
|
||||||
}
|
|
||||||
for (let url of addrs)
|
|
||||||
{
|
|
||||||
let scheme = 'http';
|
|
||||||
url = url.trim().replace(/^(https?):\/\//, (m, m1) => { scheme = m1; return ''; });
|
|
||||||
const slash = url.indexOf('/');
|
|
||||||
const colon = url.indexOf(':');
|
|
||||||
const is_local = is_local_ip[colon >= 0 ? url.substr(0, colon) : (slash >= 0 ? url.substr(0, slash) : url)];
|
|
||||||
url = scheme+'://'+(slash >= 0 ? url : url+'/v3');
|
|
||||||
if (is_local)
|
|
||||||
this.etcd_local.push(url);
|
|
||||||
else
|
|
||||||
this.etcd_urls.push(url);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async start()
|
async start()
|
||||||
|
@ -380,18 +303,7 @@ class Mon
|
||||||
await this.become_master();
|
await this.become_master();
|
||||||
await this.load_cluster_state();
|
await this.load_cluster_state();
|
||||||
await this.start_watcher(this.config.etcd_mon_retries);
|
await this.start_watcher(this.config.etcd_mon_retries);
|
||||||
for (const pool_id in this.state.config.pools)
|
|
||||||
{
|
|
||||||
if (!this.state.pool.stats[pool_id] ||
|
|
||||||
!this.state.pool.stats[pool_id].pg_real_size)
|
|
||||||
{
|
|
||||||
// Generate missing data in etcd
|
|
||||||
this.state.config.pgs.hash = null;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
await this.recheck_pgs();
|
await this.recheck_pgs();
|
||||||
this.schedule_update_stats();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async load_config()
|
async load_config()
|
||||||
|
@ -441,31 +353,6 @@ class Mon
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pick_next_etcd()
|
|
||||||
{
|
|
||||||
if (this.selected_etcd_url)
|
|
||||||
return this.selected_etcd_url;
|
|
||||||
if (!this.etcd_urls_to_try || !this.etcd_urls_to_try.length)
|
|
||||||
{
|
|
||||||
this.etcd_urls_to_try = [ ...this.etcd_local ];
|
|
||||||
const others = [ ...this.etcd_urls ];
|
|
||||||
while (others.length)
|
|
||||||
{
|
|
||||||
const url = others.splice(0|(others.length*Math.random()), 1);
|
|
||||||
this.etcd_urls_to_try.push(url[0]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
this.selected_etcd_url = this.etcd_urls_to_try.shift();
|
|
||||||
return this.selected_etcd_url;
|
|
||||||
}
|
|
||||||
|
|
||||||
restart_watcher(cur_addr)
|
|
||||||
{
|
|
||||||
if (this.selected_etcd_url == cur_addr)
|
|
||||||
this.selected_etcd_url = null;
|
|
||||||
this.start_watcher(this.config.etcd_mon_retries).catch(this.die);
|
|
||||||
}
|
|
||||||
|
|
||||||
async start_watcher(retries)
|
async start_watcher(retries)
|
||||||
{
|
{
|
||||||
let retry = 0;
|
let retry = 0;
|
||||||
|
@ -475,8 +362,7 @@ class Mon
|
||||||
}
|
}
|
||||||
while (retries < 0 || retry < retries)
|
while (retries < 0 || retry < retries)
|
||||||
{
|
{
|
||||||
const cur_addr = this.pick_next_etcd();
|
const base = 'ws'+this.etcd_urls[Math.floor(Math.random()*this.etcd_urls.length)].substr(4);
|
||||||
const base = 'ws'+cur_addr.substr(4);
|
|
||||||
const ok = await new Promise((ok, no) =>
|
const ok = await new Promise((ok, no) =>
|
||||||
{
|
{
|
||||||
const timer_id = setTimeout(() =>
|
const timer_id = setTimeout(() =>
|
||||||
|
@ -499,9 +385,9 @@ class Mon
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
if (ok)
|
if (ok)
|
||||||
|
{
|
||||||
break;
|
break;
|
||||||
if (this.selected_etcd_url == cur_addr)
|
}
|
||||||
this.selected_etcd_url = null;
|
|
||||||
this.ws = null;
|
this.ws = null;
|
||||||
retry++;
|
retry++;
|
||||||
}
|
}
|
||||||
|
@ -509,8 +395,6 @@ class Mon
|
||||||
{
|
{
|
||||||
this.die('Failed to open etcd watch websocket');
|
this.die('Failed to open etcd watch websocket');
|
||||||
}
|
}
|
||||||
const cur_addr = this.selected_etcd_url;
|
|
||||||
this.ws.on('error', () => this.restart_watcher(cur_addr));
|
|
||||||
this.ws.send(JSON.stringify({
|
this.ws.send(JSON.stringify({
|
||||||
create_request: {
|
create_request: {
|
||||||
key: b64(this.etcd_prefix+'/'),
|
key: b64(this.etcd_prefix+'/'),
|
||||||
|
@ -530,25 +414,12 @@ class Mon
|
||||||
catch (e)
|
catch (e)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
if (!data || !data.result)
|
if (!data || !data.result || !data.result.events)
|
||||||
{
|
{
|
||||||
console.error('Unknown message received from watch websocket: '+msg);
|
if (!data || !data.result || !data.result.watch_id)
|
||||||
|
{
|
||||||
|
console.error('Garbage received from watch websocket: '+msg);
|
||||||
}
|
}
|
||||||
else if (data.result.canceled)
|
|
||||||
{
|
|
||||||
// etcd watch canceled
|
|
||||||
if (data.result.compact_revision)
|
|
||||||
{
|
|
||||||
// we may miss events if we proceed
|
|
||||||
console.error('Revisions before '+data.result.compact_revision+' were compacted by etcd, exiting');
|
|
||||||
this.on_stop(1);
|
|
||||||
}
|
|
||||||
console.error('Watch canceled by etcd, reason: '+data.result.cancel_reason+', exiting');
|
|
||||||
this.on_stop(1);
|
|
||||||
}
|
|
||||||
else if (data.result.created)
|
|
||||||
{
|
|
||||||
// etcd watch created
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -581,7 +452,7 @@ class Mon
|
||||||
}
|
}
|
||||||
if (pg_states_changed)
|
if (pg_states_changed)
|
||||||
{
|
{
|
||||||
this.save_last_clean().catch(this.die);
|
this.save_last_clean().catch(console.error);
|
||||||
}
|
}
|
||||||
if (stats_changed)
|
if (stats_changed)
|
||||||
{
|
{
|
||||||
|
@ -636,7 +507,7 @@ class Mon
|
||||||
const max_ttl = this.config.etcd_mon_ttl + this.config.etcd_mon_timeout/1000*this.config.etcd_mon_retries;
|
const max_ttl = this.config.etcd_mon_ttl + this.config.etcd_mon_timeout/1000*this.config.etcd_mon_retries;
|
||||||
const res = await this.etcd_call('/lease/grant', { TTL: max_ttl }, this.config.etcd_mon_timeout, -1);
|
const res = await this.etcd_call('/lease/grant', { TTL: max_ttl }, this.config.etcd_mon_timeout, -1);
|
||||||
this.etcd_lease_id = res.ID;
|
this.etcd_lease_id = res.ID;
|
||||||
this.lease_timer = setInterval(async () =>
|
setInterval(async () =>
|
||||||
{
|
{
|
||||||
const res = await this.etcd_call('/lease/keepalive', { ID: this.etcd_lease_id }, this.config.etcd_mon_timeout, this.config.etcd_mon_retries);
|
const res = await this.etcd_call('/lease/keepalive', { ID: this.etcd_lease_id }, this.config.etcd_mon_timeout, this.config.etcd_mon_retries);
|
||||||
if (!res.result.TTL)
|
if (!res.result.TTL)
|
||||||
|
@ -644,19 +515,6 @@ class Mon
|
||||||
this.die('Lease expired');
|
this.die('Lease expired');
|
||||||
}
|
}
|
||||||
}, this.config.etcd_mon_timeout);
|
}, this.config.etcd_mon_timeout);
|
||||||
if (!this.signals_set)
|
|
||||||
{
|
|
||||||
process.on('SIGINT', this.on_stop_cb);
|
|
||||||
process.on('SIGTERM', this.on_stop_cb);
|
|
||||||
this.signals_set = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async on_stop(status)
|
|
||||||
{
|
|
||||||
clearInterval(this.lease_timer);
|
|
||||||
await this.etcd_call('/lease/revoke', { ID: this.etcd_lease_id }, this.config.etcd_mon_timeout, this.config.etcd_mon_retries);
|
|
||||||
process.exit(status);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async become_master()
|
async become_master()
|
||||||
|
@ -709,13 +567,10 @@ class Mon
|
||||||
for (const node_id in this.state.config.node_placement||{})
|
for (const node_id in this.state.config.node_placement||{})
|
||||||
{
|
{
|
||||||
const node_cfg = this.state.config.node_placement[node_id];
|
const node_cfg = this.state.config.node_placement[node_id];
|
||||||
if (/^\d+$/.exec(node_id))
|
if (!node_id || /^\d/.exec(node_id) ||
|
||||||
|
!node_cfg.level || !levels[node_cfg.level])
|
||||||
{
|
{
|
||||||
node_cfg.level = 'osd';
|
// All nodes must have non-empty non-numeric IDs and valid levels
|
||||||
}
|
|
||||||
if (!node_id || !node_cfg.level || !levels[node_cfg.level])
|
|
||||||
{
|
|
||||||
// All nodes must have non-empty IDs and valid levels
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
tree[node_id] = { id: node_id, level: node_cfg.level, parent: node_cfg.parent, children: [] };
|
tree[node_id] = { id: node_id, level: node_cfg.level, parent: node_cfg.parent, children: [] };
|
||||||
|
@ -748,10 +603,10 @@ class Mon
|
||||||
.reduce((a, c) => { a[c] = true; return a; }, {});
|
.reduce((a, c) => { a[c] = true; return a; }, {});
|
||||||
}
|
}
|
||||||
delete tree[osd_num].children;
|
delete tree[osd_num].children;
|
||||||
if (!tree[stat.host])
|
if (!tree[tree[osd_num].parent])
|
||||||
{
|
{
|
||||||
tree[stat.host] = {
|
tree[tree[osd_num].parent] = {
|
||||||
id: stat.host,
|
id: tree[osd_num].parent,
|
||||||
level: 'host',
|
level: 'host',
|
||||||
parent: null,
|
parent: null,
|
||||||
children: [],
|
children: [],
|
||||||
|
@ -1097,7 +952,6 @@ class Mon
|
||||||
pg_size: pool_cfg.pg_size,
|
pg_size: pool_cfg.pg_size,
|
||||||
pg_minsize: pool_cfg.pg_minsize,
|
pg_minsize: pool_cfg.pg_minsize,
|
||||||
max_combinations: pool_cfg.max_osd_combinations,
|
max_combinations: pool_cfg.max_osd_combinations,
|
||||||
round_robin: pool_cfg.scheme != 'replicated',
|
|
||||||
};
|
};
|
||||||
let optimize_result;
|
let optimize_result;
|
||||||
if (old_pg_count > 0)
|
if (old_pg_count > 0)
|
||||||
|
@ -1155,19 +1009,6 @@ class Mon
|
||||||
} });
|
} });
|
||||||
}
|
}
|
||||||
LPOptimizer.print_change_stats(optimize_result);
|
LPOptimizer.print_change_stats(optimize_result);
|
||||||
const pg_effsize = Math.min(pool_cfg.pg_size, Object.keys(pool_tree).length);
|
|
||||||
this.state.pool.stats[pool_id] = {
|
|
||||||
used_raw_tb: (this.state.pool.stats[pool_id]||{}).used_raw_tb || 0,
|
|
||||||
total_raw_tb: optimize_result.space,
|
|
||||||
pg_real_size: pg_effsize,
|
|
||||||
raw_to_usable: pg_effsize / (pool_cfg.scheme === 'replicated'
|
|
||||||
? 1 : (pool_cfg.pg_size - (pool_cfg.parity_chunks||0))),
|
|
||||||
space_efficiency: optimize_result.space/(optimize_result.total_space||1),
|
|
||||||
};
|
|
||||||
etcd_request.success.push({ requestPut: {
|
|
||||||
key: b64(this.etcd_prefix+'/pool/stats/'+pool_id),
|
|
||||||
value: b64(JSON.stringify(this.state.pool.stats[pool_id])),
|
|
||||||
} });
|
|
||||||
this.save_new_pgs_txn(etcd_request, pool_id, up_osds, real_prev_pgs, optimize_result.int_pgs, pg_history);
|
this.save_new_pgs_txn(etcd_request, pool_id, up_osds, real_prev_pgs, optimize_result.int_pgs, pg_history);
|
||||||
}
|
}
|
||||||
this.state.config.pgs.hash = tree_hash;
|
this.state.config.pgs.hash = tree_hash;
|
||||||
|
@ -1270,11 +1111,11 @@ class Mon
|
||||||
this.recheck_timer = setTimeout(() =>
|
this.recheck_timer = setTimeout(() =>
|
||||||
{
|
{
|
||||||
this.recheck_timer = null;
|
this.recheck_timer = null;
|
||||||
this.recheck_pgs().catch(this.die);
|
this.recheck_pgs().catch(console.error);
|
||||||
}, this.config.mon_change_timeout || 1000);
|
}, this.config.mon_change_timeout || 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
sum_op_stats(timestamp, prev_stats)
|
sum_stats()
|
||||||
{
|
{
|
||||||
const op_stats = {}, subop_stats = {}, recovery_stats = {};
|
const op_stats = {}, subop_stats = {}, recovery_stats = {};
|
||||||
for (const osd in this.state.osd.stats)
|
for (const osd in this.state.osd.stats)
|
||||||
|
@ -1300,29 +1141,6 @@ class Mon
|
||||||
recovery_stats[op].bytes += BigInt(st.recovery_stats[op].bytes||0);
|
recovery_stats[op].bytes += BigInt(st.recovery_stats[op].bytes||0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (prev_stats && prev_stats.timestamp >= timestamp)
|
|
||||||
{
|
|
||||||
prev_stats = null;
|
|
||||||
}
|
|
||||||
const tm = prev_stats ? BigInt(timestamp - prev_stats.timestamp) : 0;
|
|
||||||
for (const op in op_stats)
|
|
||||||
{
|
|
||||||
op_stats[op].bps = prev_stats ? (op_stats[op].bytes - prev_stats.op_stats[op].bytes) * 1000n / tm : 0;
|
|
||||||
op_stats[op].iops = prev_stats ? (op_stats[op].count - prev_stats.op_stats[op].count) * 1000n / tm : 0;
|
|
||||||
op_stats[op].lat = prev_stats ? (op_stats[op].usec - prev_stats.op_stats[op].usec)
|
|
||||||
/ ((op_stats[op].count - prev_stats.op_stats[op].count) || 1n) : 0;
|
|
||||||
}
|
|
||||||
for (const op in subop_stats)
|
|
||||||
{
|
|
||||||
subop_stats[op].iops = prev_stats ? (subop_stats[op].count - prev_stats.subop_stats[op].count) * 1000n / tm : 0;
|
|
||||||
subop_stats[op].lat = prev_stats ? (subop_stats[op].usec - prev_stats.subop_stats[op].usec)
|
|
||||||
/ ((subop_stats[op].count - prev_stats.subop_stats[op].count) || 1n) : 0;
|
|
||||||
}
|
|
||||||
for (const op in recovery_stats)
|
|
||||||
{
|
|
||||||
recovery_stats[op].bps = prev_stats ? (recovery_stats[op].bytes - prev_stats.recovery_stats[op].bytes) * 1000n / tm : 0;
|
|
||||||
recovery_stats[op].iops = prev_stats ? (recovery_stats[op].count - prev_stats.recovery_stats[op].count) * 1000n / tm : 0;
|
|
||||||
}
|
|
||||||
return { op_stats, subop_stats, recovery_stats };
|
return { op_stats, subop_stats, recovery_stats };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1349,7 +1167,7 @@ class Mon
|
||||||
return object_counts;
|
return object_counts;
|
||||||
}
|
}
|
||||||
|
|
||||||
sum_inode_stats(prev_stats, timestamp, prev_timestamp)
|
sum_inode_stats()
|
||||||
{
|
{
|
||||||
const inode_stats = {};
|
const inode_stats = {};
|
||||||
const inode_stub = () => ({
|
const inode_stub = () => ({
|
||||||
|
@ -1358,38 +1176,18 @@ class Mon
|
||||||
write: { count: 0n, usec: 0n, bytes: 0n },
|
write: { count: 0n, usec: 0n, bytes: 0n },
|
||||||
delete: { count: 0n, usec: 0n, bytes: 0n },
|
delete: { count: 0n, usec: 0n, bytes: 0n },
|
||||||
});
|
});
|
||||||
const seen_pools = {};
|
|
||||||
for (const pool_id in this.state.config.pools)
|
|
||||||
{
|
|
||||||
seen_pools[pool_id] = true;
|
|
||||||
this.state.pool.stats[pool_id] = this.state.pool.stats[pool_id] || {};
|
|
||||||
this.state.pool.stats[pool_id].used_raw_tb = 0n;
|
|
||||||
}
|
|
||||||
for (const osd_num in this.state.osd.space)
|
for (const osd_num in this.state.osd.space)
|
||||||
{
|
{
|
||||||
for (const pool_id in this.state.osd.space[osd_num])
|
for (const pool_id in this.state.osd.space[osd_num])
|
||||||
{
|
{
|
||||||
this.state.pool.stats[pool_id] = this.state.pool.stats[pool_id] || {};
|
|
||||||
if (!seen_pools[pool_id])
|
|
||||||
{
|
|
||||||
this.state.pool.stats[pool_id].used_raw_tb = 0n;
|
|
||||||
seen_pools[pool_id] = true;
|
|
||||||
}
|
|
||||||
inode_stats[pool_id] = inode_stats[pool_id] || {};
|
inode_stats[pool_id] = inode_stats[pool_id] || {};
|
||||||
for (const inode_num in this.state.osd.space[osd_num][pool_id])
|
for (const inode_num in this.state.osd.space[osd_num][pool_id])
|
||||||
{
|
{
|
||||||
const u = BigInt(this.state.osd.space[osd_num][pool_id][inode_num]||0);
|
|
||||||
inode_stats[pool_id][inode_num] = inode_stats[pool_id][inode_num] || inode_stub();
|
inode_stats[pool_id][inode_num] = inode_stats[pool_id][inode_num] || inode_stub();
|
||||||
inode_stats[pool_id][inode_num].raw_used += u;
|
inode_stats[pool_id][inode_num].raw_used += BigInt(this.state.osd.space[osd_num][pool_id][inode_num]||0);
|
||||||
this.state.pool.stats[pool_id].used_raw_tb += u;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (const pool_id in seen_pools)
|
|
||||||
{
|
|
||||||
const used = this.state.pool.stats[pool_id].used_raw_tb;
|
|
||||||
this.state.pool.stats[pool_id].used_raw_tb = Number(used)/1024/1024/1024/1024;
|
|
||||||
}
|
|
||||||
for (const osd_num in this.state.osd.inodestats)
|
for (const osd_num in this.state.osd.inodestats)
|
||||||
{
|
{
|
||||||
const ist = this.state.osd.inodestats[osd_num];
|
const ist = this.state.osd.inodestats[osd_num];
|
||||||
|
@ -1408,39 +1206,43 @@ class Mon
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (prev_stats && prev_timestamp >= timestamp)
|
|
||||||
{
|
|
||||||
prev_stats = null;
|
|
||||||
}
|
|
||||||
const tm = prev_stats ? BigInt(timestamp - prev_timestamp) : 0;
|
|
||||||
for (const pool_id in inode_stats)
|
|
||||||
{
|
|
||||||
for (const inode_num in inode_stats[pool_id])
|
|
||||||
{
|
|
||||||
let nonzero = inode_stats[pool_id][inode_num].raw_used > 0;
|
|
||||||
for (const op of [ 'read', 'write', 'delete' ])
|
|
||||||
{
|
|
||||||
const op_st = inode_stats[pool_id][inode_num][op];
|
|
||||||
const prev_st = prev_stats && prev_stats[pool_id] && prev_stats[pool_id][inode_num] && prev_stats[pool_id][inode_num][op];
|
|
||||||
op_st.bps = prev_st ? (op_st.bytes - prev_st.bytes) * 1000n / tm : 0;
|
|
||||||
op_st.iops = prev_st ? (op_st.count - prev_st.count) * 1000n / tm : 0;
|
|
||||||
op_st.lat = prev_st ? (op_st.usec - prev_st.usec) / ((op_st.count - prev_st.count) || 1n) : 0;
|
|
||||||
if (op_st.bps > 0 || op_st.iops > 0 || op_st.lat > 0)
|
|
||||||
nonzero = true;
|
|
||||||
}
|
|
||||||
if (!nonzero && (!this.state.config.inode[pool_id] || !this.state.config.inode[pool_id][inode_num]))
|
|
||||||
{
|
|
||||||
// Deleted inode (no data, no I/O, no config)
|
|
||||||
delete inode_stats[pool_id][inode_num];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return inode_stats;
|
return inode_stats;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fix_stat_overflows(obj, scratch)
|
||||||
|
{
|
||||||
|
for (const k in obj)
|
||||||
|
{
|
||||||
|
if (typeof obj[k] == 'bigint')
|
||||||
|
{
|
||||||
|
if (obj[k] >= 0x10000000000000000n)
|
||||||
|
{
|
||||||
|
if (scratch[k])
|
||||||
|
{
|
||||||
|
for (const k2 in scratch)
|
||||||
|
{
|
||||||
|
obj[k2] -= scratch[k2];
|
||||||
|
scratch[k2] = 0n;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (const k2 in obj)
|
||||||
|
{
|
||||||
|
scratch[k2] = obj[k2];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (typeof obj[k] == 'object')
|
||||||
|
{
|
||||||
|
this.fix_stat_overflows(obj[k], scratch[k] = (scratch[k] || {}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
serialize_bigints(obj)
|
serialize_bigints(obj)
|
||||||
{
|
{
|
||||||
obj = { ...obj };
|
|
||||||
for (const k in obj)
|
for (const k in obj)
|
||||||
{
|
{
|
||||||
if (typeof obj[k] == 'bigint')
|
if (typeof obj[k] == 'bigint')
|
||||||
|
@ -1449,26 +1251,22 @@ class Mon
|
||||||
}
|
}
|
||||||
else if (typeof obj[k] == 'object')
|
else if (typeof obj[k] == 'object')
|
||||||
{
|
{
|
||||||
obj[k] = this.serialize_bigints(obj[k]);
|
this.serialize_bigints(obj[k]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return obj;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async update_total_stats()
|
async update_total_stats()
|
||||||
{
|
{
|
||||||
const txn = [];
|
const txn = [];
|
||||||
const timestamp = Date.now();
|
const stats = this.sum_stats();
|
||||||
const object_counts = this.sum_object_counts();
|
const object_counts = this.sum_object_counts();
|
||||||
let stats = this.sum_op_stats(timestamp, this.prev_stats);
|
const inode_stats = this.sum_inode_stats();
|
||||||
let inode_stats = this.sum_inode_stats(
|
this.fix_stat_overflows(stats, (this.prev_stats = this.prev_stats || {}));
|
||||||
this.prev_stats ? this.prev_stats.inode_stats : null,
|
this.fix_stat_overflows(inode_stats, (this.prev_inode_stats = this.prev_inode_stats || {}));
|
||||||
timestamp, this.prev_stats ? this.prev_stats.timestamp : null
|
|
||||||
);
|
|
||||||
this.prev_stats = { timestamp, ...stats, inode_stats };
|
|
||||||
stats.object_counts = object_counts;
|
stats.object_counts = object_counts;
|
||||||
stats = this.serialize_bigints(stats);
|
this.serialize_bigints(stats);
|
||||||
inode_stats = this.serialize_bigints(inode_stats);
|
this.serialize_bigints(inode_stats);
|
||||||
txn.push({ requestPut: { key: b64(this.etcd_prefix+'/stats'), value: b64(JSON.stringify(stats)) } });
|
txn.push({ requestPut: { key: b64(this.etcd_prefix+'/stats'), value: b64(JSON.stringify(stats)) } });
|
||||||
for (const pool_id in inode_stats)
|
for (const pool_id in inode_stats)
|
||||||
{
|
{
|
||||||
|
@ -1480,27 +1278,6 @@ class Mon
|
||||||
} });
|
} });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (const pool_id in this.state.inode.stats)
|
|
||||||
{
|
|
||||||
for (const inode_num in this.state.inode.stats[pool_id])
|
|
||||||
{
|
|
||||||
if (!inode_stats[pool_id] || !inode_stats[pool_id][inode_num])
|
|
||||||
{
|
|
||||||
txn.push({ requestDeleteRange: {
|
|
||||||
key: b64(this.etcd_prefix+'/inode/stats/'+pool_id+'/'+inode_num),
|
|
||||||
} });
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (const pool_id in this.state.pool.stats)
|
|
||||||
{
|
|
||||||
const pool_stats = { ...this.state.pool.stats[pool_id] };
|
|
||||||
this.serialize_bigints(pool_stats);
|
|
||||||
txn.push({ requestPut: {
|
|
||||||
key: b64(this.etcd_prefix+'/pool/stats/'+pool_id),
|
|
||||||
value: b64(JSON.stringify(pool_stats)),
|
|
||||||
} });
|
|
||||||
}
|
|
||||||
if (txn.length)
|
if (txn.length)
|
||||||
{
|
{
|
||||||
await this.etcd_call('/kv/txn', { success: txn }, this.config.etcd_mon_timeout, 0);
|
await this.etcd_call('/kv/txn', { success: txn }, this.config.etcd_mon_timeout, 0);
|
||||||
|
@ -1511,13 +1288,14 @@ class Mon
|
||||||
{
|
{
|
||||||
if (this.stats_timer)
|
if (this.stats_timer)
|
||||||
{
|
{
|
||||||
return;
|
clearTimeout(this.stats_timer);
|
||||||
|
this.stats_timer = null;
|
||||||
}
|
}
|
||||||
this.stats_timer = setTimeout(() =>
|
this.stats_timer = setTimeout(() =>
|
||||||
{
|
{
|
||||||
this.stats_timer = null;
|
this.stats_timer = null;
|
||||||
this.update_total_stats().catch(console.error);
|
this.update_total_stats().catch(console.error);
|
||||||
}, this.config.mon_stats_timeout);
|
}, this.config.mon_stats_timeout || 1000);
|
||||||
}
|
}
|
||||||
|
|
||||||
parse_kv(kv)
|
parse_kv(kv)
|
||||||
|
@ -1557,7 +1335,7 @@ class Mon
|
||||||
cur[key_parts[key_parts.length-1]] = kv.value;
|
cur[key_parts[key_parts.length-1]] = kv.value;
|
||||||
if (key === 'config/global')
|
if (key === 'config/global')
|
||||||
{
|
{
|
||||||
this.config = { ...this.initConfig, ...this.state.config.global };
|
this.config = this.state.config.global;
|
||||||
this.check_config();
|
this.check_config();
|
||||||
for (const osd_num in this.state.osd.stats)
|
for (const osd_num in this.state.osd.stats)
|
||||||
{
|
{
|
||||||
|
@ -1594,15 +1372,12 @@ class Mon
|
||||||
}
|
}
|
||||||
while (retries < 0 || retry < retries)
|
while (retries < 0 || retry < retries)
|
||||||
{
|
{
|
||||||
retry++;
|
const base = this.etcd_urls[Math.floor(Math.random()*this.etcd_urls.length)];
|
||||||
const base = this.pick_next_etcd();
|
|
||||||
const res = await POST(base+path, body, timeout);
|
const res = await POST(base+path, body, timeout);
|
||||||
if (res.error)
|
if (res.error)
|
||||||
{
|
{
|
||||||
if (this.selected_etcd_url == base)
|
console.error('etcd returned error: '+res.error);
|
||||||
this.selected_etcd_url = null;
|
break;
|
||||||
console.error('failed to query etcd: '+res.error);
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
if (res.json)
|
if (res.json)
|
||||||
{
|
{
|
||||||
|
@ -1611,20 +1386,26 @@ class Mon
|
||||||
console.error('etcd returned error: '+res.json.error);
|
console.error('etcd returned error: '+res.json.error);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
if (this.etcd_urls.length > 1)
|
||||||
|
{
|
||||||
|
// Stick to the same etcd for the rest of calls
|
||||||
|
this.etcd_urls = [ base ];
|
||||||
|
}
|
||||||
return res.json;
|
return res.json;
|
||||||
}
|
}
|
||||||
|
retry++;
|
||||||
}
|
}
|
||||||
this.die();
|
this.die();
|
||||||
}
|
}
|
||||||
|
|
||||||
_die(err)
|
die(err)
|
||||||
{
|
{
|
||||||
// In fact we can just try to rejoin
|
// In fact we can just try to rejoin
|
||||||
console.error(new Error(err || 'Cluster connection failed'));
|
console.error(new Error(err || 'Cluster connection failed'));
|
||||||
process.exit(1);
|
process.exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
local_ips(all)
|
local_ips()
|
||||||
{
|
{
|
||||||
const ips = [];
|
const ips = [];
|
||||||
const ifaces = os.networkInterfaces();
|
const ifaces = os.networkInterfaces();
|
||||||
|
@ -1632,7 +1413,7 @@ class Mon
|
||||||
{
|
{
|
||||||
for (const iface of ifaces[ifname])
|
for (const iface of ifaces[ifname])
|
||||||
{
|
{
|
||||||
if (iface.family == 'IPv4' && !iface.internal || all)
|
if (iface.family == 'IPv4' && !iface.internal)
|
||||||
{
|
{
|
||||||
ips.push(iface.address);
|
ips.push(iface.address);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,33 +0,0 @@
|
||||||
// Copyright (c) Vitaliy Filippov, 2019+
|
|
||||||
// License: VNPL-1.1 (see README.md for details)
|
|
||||||
|
|
||||||
const LPOptimizer = require('./lp-optimizer.js');
|
|
||||||
|
|
||||||
const osd_tree = {
|
|
||||||
100: {
|
|
||||||
1: 0.1,
|
|
||||||
2: 0.1,
|
|
||||||
3: 0.1,
|
|
||||||
},
|
|
||||||
200: {
|
|
||||||
4: 0.1,
|
|
||||||
5: 0.1,
|
|
||||||
6: 0.1,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
async function run()
|
|
||||||
{
|
|
||||||
let res;
|
|
||||||
console.log('256 PGs, 3+3 OSDs, size=2');
|
|
||||||
res = await LPOptimizer.optimize_initial({ osd_tree, pg_size: 2, pg_count: 256 });
|
|
||||||
LPOptimizer.print_change_stats(res, false);
|
|
||||||
|
|
||||||
// Should NOT fail with the "unfeasible or unbounded" exception
|
|
||||||
console.log('\nRemoving osd.2');
|
|
||||||
delete osd_tree[100][2];
|
|
||||||
res = await LPOptimizer.optimize_change({ prev_pgs: res.int_pgs, osd_tree, pg_size: 2 });
|
|
||||||
LPOptimizer.print_change_stats(res, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
run().catch(console.error);
|
|
|
@ -1,503 +0,0 @@
|
||||||
# Install as /usr/share/perl5/PVE/Storage/Custom/VitastorPlugin.pm
|
|
||||||
|
|
||||||
# Proxmox Vitastor Driver
|
|
||||||
# Copyright (c) Vitaliy Filippov, 2021+
|
|
||||||
# License: VNPL-1.1 or GNU AGPLv3.0
|
|
||||||
|
|
||||||
package PVE::Storage::Custom::VitastorPlugin;
|
|
||||||
|
|
||||||
use strict;
|
|
||||||
use warnings;
|
|
||||||
|
|
||||||
use JSON;
|
|
||||||
|
|
||||||
use PVE::Storage::Plugin;
|
|
||||||
use PVE::Tools qw(run_command);
|
|
||||||
|
|
||||||
use base qw(PVE::Storage::Plugin);
|
|
||||||
|
|
||||||
sub api
|
|
||||||
{
|
|
||||||
# Trick it :)
|
|
||||||
return PVE::Storage->APIVER;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub run_cli
|
|
||||||
{
|
|
||||||
my ($scfg, $cmd, %args) = @_;
|
|
||||||
my $retval;
|
|
||||||
my $stderr = '';
|
|
||||||
my $errmsg = $args{errmsg} ? $args{errmsg}.": " : "vitastor-cli error: ";
|
|
||||||
my $json = delete $args{json};
|
|
||||||
$json = 1 if !defined $json;
|
|
||||||
my $binary = delete $args{binary};
|
|
||||||
$binary = '/usr/bin/vitastor-cli' if !defined $binary;
|
|
||||||
if (!exists($args{errfunc}))
|
|
||||||
{
|
|
||||||
$args{errfunc} = sub
|
|
||||||
{
|
|
||||||
my $line = shift;
|
|
||||||
print STDERR $line;
|
|
||||||
*STDERR->flush();
|
|
||||||
$stderr .= $line;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
if (!exists($args{outfunc}))
|
|
||||||
{
|
|
||||||
$retval = '';
|
|
||||||
$args{outfunc} = sub { $retval .= shift };
|
|
||||||
if ($json)
|
|
||||||
{
|
|
||||||
unshift @$cmd, '--json';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if ($scfg->{vitastor_etcd_address})
|
|
||||||
{
|
|
||||||
unshift @$cmd, '--etcd_address', $scfg->{vitastor_etcd_address};
|
|
||||||
}
|
|
||||||
if ($scfg->{vitastor_config_path})
|
|
||||||
{
|
|
||||||
unshift @$cmd, '--config_path', $scfg->{vitastor_config_path};
|
|
||||||
}
|
|
||||||
unshift @$cmd, $binary;
|
|
||||||
eval { run_command($cmd, %args); };
|
|
||||||
if (my $err = $@)
|
|
||||||
{
|
|
||||||
die "Error invoking vitastor-cli: $err";
|
|
||||||
}
|
|
||||||
if (defined $retval)
|
|
||||||
{
|
|
||||||
# untaint
|
|
||||||
$retval =~ /^(.*)$/s;
|
|
||||||
if ($json)
|
|
||||||
{
|
|
||||||
eval { $retval = JSON::decode_json($1); };
|
|
||||||
if ($@)
|
|
||||||
{
|
|
||||||
die "vitastor-cli returned bad JSON: $@";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$retval = $1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return $retval;
|
|
||||||
}
|
|
||||||
|
|
||||||
# Configuration
|
|
||||||
|
|
||||||
sub type
|
|
||||||
{
|
|
||||||
return 'vitastor';
|
|
||||||
}
|
|
||||||
|
|
||||||
sub plugindata
|
|
||||||
{
|
|
||||||
return {
|
|
||||||
content => [ { images => 1, rootdir => 1 }, { images => 1 } ],
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
sub properties
|
|
||||||
{
|
|
||||||
return {
|
|
||||||
vitastor_etcd_address => {
|
|
||||||
description => 'IP address(es) of etcd.',
|
|
||||||
type => 'string',
|
|
||||||
format => 'pve-storage-portal-dns-list',
|
|
||||||
},
|
|
||||||
vitastor_etcd_prefix => {
|
|
||||||
description => 'Prefix for Vitastor etcd metadata',
|
|
||||||
type => 'string',
|
|
||||||
},
|
|
||||||
vitastor_config_path => {
|
|
||||||
description => 'Path to Vitastor configuration file',
|
|
||||||
type => 'string',
|
|
||||||
},
|
|
||||||
vitastor_prefix => {
|
|
||||||
description => 'Image name prefix',
|
|
||||||
type => 'string',
|
|
||||||
},
|
|
||||||
vitastor_pool => {
|
|
||||||
description => 'Default pool to use for images',
|
|
||||||
type => 'string',
|
|
||||||
},
|
|
||||||
vitastor_nbd => {
|
|
||||||
description => 'Use kernel NBD devices (slower)',
|
|
||||||
type => 'boolean',
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
sub options
|
|
||||||
{
|
|
||||||
return {
|
|
||||||
nodes => { optional => 1 },
|
|
||||||
disable => { optional => 1 },
|
|
||||||
vitastor_etcd_address => { optional => 1},
|
|
||||||
vitastor_etcd_prefix => { optional => 1 },
|
|
||||||
vitastor_config_path => { optional => 1 },
|
|
||||||
vitastor_prefix => { optional => 1 },
|
|
||||||
vitastor_pool => {},
|
|
||||||
vitastor_nbd => { optional => 1 },
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
# Storage implementation
|
|
||||||
|
|
||||||
sub parse_volname
|
|
||||||
{
|
|
||||||
my ($class, $volname) = @_;
|
|
||||||
if ($volname =~ m/^((base-(\d+)-\S+)\/)?((?:(base)|(vm))-(\d+)-\S+)$/)
|
|
||||||
{
|
|
||||||
# ($vtype, $name, $vmid, $basename, $basevmid, $isBase, $format)
|
|
||||||
return ('images', $4, $7, $2, $3, $5, 'raw');
|
|
||||||
}
|
|
||||||
die "unable to parse vitastor volume name '$volname'\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _qemu_option
|
|
||||||
{
|
|
||||||
my ($k, $v) = @_;
|
|
||||||
if (defined $v && $v ne "")
|
|
||||||
{
|
|
||||||
$v =~ s/:/\\:/gso;
|
|
||||||
return ":$k=$v";
|
|
||||||
}
|
|
||||||
return "";
|
|
||||||
}
|
|
||||||
|
|
||||||
sub path
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $volname, $storeid, $snapname) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my ($vtype, $name, $vmid) = $class->parse_volname($volname);
|
|
||||||
$name .= '@'.$snapname if $snapname;
|
|
||||||
if ($scfg->{vitastor_nbd})
|
|
||||||
{
|
|
||||||
my $mapped = run_cli($scfg, [ 'ls' ], binary => '/usr/bin/vitastor-nbd');
|
|
||||||
my ($kerneldev) = grep { $mapped->{$_}->{image} eq $prefix.$name } keys %$mapped;
|
|
||||||
die "Image not mapped via NBD" if !$kerneldev;
|
|
||||||
return ($kerneldev, $vmid, $vtype);
|
|
||||||
}
|
|
||||||
my $path = "vitastor";
|
|
||||||
$path .= _qemu_option('config_path', $scfg->{vitastor_config_path});
|
|
||||||
# FIXME This is the only exception: etcd_address -> etcd_host for qemu
|
|
||||||
$path .= _qemu_option('etcd_host', $scfg->{vitastor_etcd_address});
|
|
||||||
$path .= _qemu_option('etcd_prefix', $scfg->{vitastor_etcd_prefix});
|
|
||||||
$path .= _qemu_option('image', $prefix.$name);
|
|
||||||
return ($path, $vmid, $vtype);
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _find_free_diskname
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $vmid, $fmt, $add_fmt_suffix) = @_;
|
|
||||||
my $list = _process_list($scfg, $storeid, run_cli($scfg, [ 'ls' ]));
|
|
||||||
$list = [ map { $_->{name} } @$list ];
|
|
||||||
return PVE::Storage::Plugin::get_next_vm_diskname($list, $storeid, $vmid, undef, $scfg);
|
|
||||||
}
|
|
||||||
|
|
||||||
# Used only in "Create Template" and, in fact, converts a VM into a template
|
|
||||||
# As a consequence, this is always invoked with the VM powered off
|
|
||||||
# So we just rename vm-xxx to base-xxx and make it a readonly base layer
|
|
||||||
sub create_base
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $volname) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
|
|
||||||
my ($vtype, $name, $vmid, $basename, $basevmid, $isBase) = $class->parse_volname($volname);
|
|
||||||
die "create_base not possible with base image\n" if $isBase;
|
|
||||||
|
|
||||||
my $info = _process_list($scfg, $storeid, run_cli($scfg, [ 'ls', $prefix.$name ]))->[0];
|
|
||||||
die "image $name does not exist\n" if !$info;
|
|
||||||
|
|
||||||
die "volname '$volname' contains wrong information about parent {$info->{parent}} $basename\n"
|
|
||||||
if $basename && (!$info->{parent} || $info->{parent} ne $basename);
|
|
||||||
|
|
||||||
my $newname = $name;
|
|
||||||
$newname =~ s/^vm-/base-/;
|
|
||||||
|
|
||||||
my $newvolname = $basename ? "$basename/$newname" : "$newname";
|
|
||||||
run_cli($scfg, [ 'modify', '--rename', $prefix.$newname, '--readonly', $prefix.$name ], json => 0);
|
|
||||||
|
|
||||||
return $newvolname;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub clone_image
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $storeid, $volname, $vmid, $snapname) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
|
|
||||||
my $snap = '';
|
|
||||||
$snap = '@'.$snapname if length $snapname;
|
|
||||||
|
|
||||||
my ($vtype, $basename, $basevmid, undef, undef, $isBase) = $class->parse_volname($volname);
|
|
||||||
die "$volname is not a base image and snapname is not provided\n" if !$isBase && !length($snapname);
|
|
||||||
|
|
||||||
my $name = $class->find_free_diskname($storeid, $scfg, $vmid);
|
|
||||||
|
|
||||||
warn "clone $volname: $basename snapname $snap to $name\n";
|
|
||||||
|
|
||||||
my $newvol = "$basename/$name";
|
|
||||||
$newvol = $name if length($snapname);
|
|
||||||
|
|
||||||
run_cli($scfg, [ 'create', '--parent', $prefix.$basename.$snap, $prefix.$name ], json => 0);
|
|
||||||
|
|
||||||
return $newvol;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub alloc_image
|
|
||||||
{
|
|
||||||
# $size is in kb in this method
|
|
||||||
my ($class, $storeid, $scfg, $vmid, $fmt, $name, $size) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
die "illegal name '$name' - should be 'vm-$vmid-*'\n" if $name && $name !~ m/^vm-$vmid-/;
|
|
||||||
$name = $class->find_free_diskname($storeid, $scfg, $vmid) if !$name;
|
|
||||||
run_cli($scfg, [ 'create', '--size', (int(($size+3)/4)*4).'k', '--pool', $scfg->{vitastor_pool}, $prefix.$name ], json => 0);
|
|
||||||
return $name;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub free_image
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $volname, $isBase) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my ($vtype, $name, $vmid, undef, undef, undef) = $class->parse_volname($volname);
|
|
||||||
$class->deactivate_volume($storeid, $scfg, $volname);
|
|
||||||
my $full_list = run_cli($scfg, [ 'ls', '-l' ]);
|
|
||||||
my $list = _process_list($scfg, $storeid, $full_list);
|
|
||||||
# Remove image and all its snapshots
|
|
||||||
my $rm_names = {
|
|
||||||
map { ($prefix.$_->{name} => 1) }
|
|
||||||
grep { $_->{name} eq $name || substr($_->{name}, 0, length($name)+1) eq ($name.'@') }
|
|
||||||
@$list
|
|
||||||
};
|
|
||||||
my $children = [ grep { $_->{parent_name} && $rm_names->{$_->{parent_name}} } @$full_list ];
|
|
||||||
die "Image has children: ".join(', ', map {
|
|
||||||
substr($_->{name}, 0, length $prefix) eq $prefix
|
|
||||||
? substr($_->name, length $prefix)
|
|
||||||
: $_->{name}
|
|
||||||
} @$children)."\n" if @$children;
|
|
||||||
my $to_remove = [ grep { $rm_names->{$_->{name}} } @$full_list ];
|
|
||||||
for my $rmi (@$to_remove)
|
|
||||||
{
|
|
||||||
run_cli($scfg, [ 'rm-data', '--pool', $rmi->{pool_id}, '--inode', $rmi->{inode_num} ], json => 0);
|
|
||||||
}
|
|
||||||
for my $rmi (@$to_remove)
|
|
||||||
{
|
|
||||||
run_cli($scfg, [ 'rm', $rmi->{name} ], json => 0);
|
|
||||||
}
|
|
||||||
return undef;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub _process_list
|
|
||||||
{
|
|
||||||
my ($scfg, $storeid, $result) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my $list = [];
|
|
||||||
foreach my $el (@$result)
|
|
||||||
{
|
|
||||||
next if !$el->{name} || length($prefix) && substr($el->{name}, 0, length $prefix) ne $prefix;
|
|
||||||
my $name = substr($el->{name}, length $prefix);
|
|
||||||
next if $name =~ /@/;
|
|
||||||
my ($owner) = $name =~ /^(?:vm|base)-(\d+)-/s;
|
|
||||||
next if !defined $owner;
|
|
||||||
my $parent = !defined $el->{parent_name}
|
|
||||||
? undef
|
|
||||||
: ($prefix eq '' || substr($el->{parent_name}, 0, length $prefix) eq $prefix
|
|
||||||
? substr($el->{parent_name}, length $prefix) : '');
|
|
||||||
my $volid = $parent && $parent =~ /^(base-\d+-\S+)$/s
|
|
||||||
? "$storeid:$1/$name" : "$storeid:$name";
|
|
||||||
push @$list, {
|
|
||||||
format => 'raw',
|
|
||||||
volid => $volid,
|
|
||||||
name => $name,
|
|
||||||
size => $el->{size},
|
|
||||||
parent => $parent,
|
|
||||||
vmid => $owner,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return $list;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub list_images
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $vmid, $vollist, $cache) = @_;
|
|
||||||
my $list = _process_list($scfg, $storeid, run_cli($scfg, [ 'ls', '-l' ]));
|
|
||||||
if ($vollist)
|
|
||||||
{
|
|
||||||
my $h = { map { ($_ => 1) } @$vollist };
|
|
||||||
$list = [ grep { $h->{$_->{volid}} } @$list ]
|
|
||||||
}
|
|
||||||
elsif (defined $vmid)
|
|
||||||
{
|
|
||||||
$list = [ grep { $_->{vmid} eq $vmid } @$list ];
|
|
||||||
}
|
|
||||||
return $list;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub status
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $cache) = @_;
|
|
||||||
my $stats = [ grep { $_->{name} eq $scfg->{vitastor_pool} } @{ run_cli($scfg, [ 'df' ]) } ]->[0];
|
|
||||||
my $free = $stats ? $stats->{max_available} : 0;
|
|
||||||
my $used = $stats ? $stats->{used_raw}/($stats->{raw_to_usable}||1) : 0;
|
|
||||||
my $total = $free+$used;
|
|
||||||
my $active = $stats ? 1 : 0;
|
|
||||||
return ($total, $free, $used, $active);
|
|
||||||
}
|
|
||||||
|
|
||||||
sub activate_storage
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $cache) = @_;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub deactivate_storage
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $cache) = @_;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub map_volume
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $volname, $snapname) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
|
|
||||||
my ($vtype, $img_name, $vmid) = $class->parse_volname($volname);
|
|
||||||
my $name = $img_name;
|
|
||||||
$name .= '@'.$snapname if $snapname;
|
|
||||||
|
|
||||||
my $mapped = run_cli($scfg, [ 'ls' ], binary => '/usr/bin/vitastor-nbd');
|
|
||||||
my ($kerneldev) = grep { $mapped->{$_}->{image} eq $prefix.$name } keys %$mapped;
|
|
||||||
return $kerneldev if $kerneldev && -b $kerneldev; # already mapped
|
|
||||||
|
|
||||||
$kerneldev = run_cli($scfg, [ 'map', '--image', $prefix.$name ], binary => '/usr/bin/vitastor-nbd', json => 0);
|
|
||||||
return $kerneldev;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub unmap_volume
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $volname, $snapname) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
|
|
||||||
return 1 if !$scfg->{vitastor_nbd};
|
|
||||||
|
|
||||||
my ($vtype, $name, $vmid) = $class->parse_volname($volname);
|
|
||||||
$name .= '@'.$snapname if $snapname;
|
|
||||||
|
|
||||||
my $mapped = run_cli($scfg, [ 'ls' ], binary => '/usr/bin/vitastor-nbd');
|
|
||||||
my ($kerneldev) = grep { $mapped->{$_}->{image} eq $prefix.$name } keys %$mapped;
|
|
||||||
if ($kerneldev && -b $kerneldev)
|
|
||||||
{
|
|
||||||
run_cli($scfg, [ 'unmap', $kerneldev ], binary => '/usr/bin/vitastor-nbd', json => 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub activate_volume
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $volname, $snapname, $cache) = @_;
|
|
||||||
$class->map_volume($storeid, $scfg, $volname, $snapname) if $scfg->{vitastor_nbd};
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub deactivate_volume
|
|
||||||
{
|
|
||||||
my ($class, $storeid, $scfg, $volname, $snapname, $cache) = @_;
|
|
||||||
$class->unmap_volume($storeid, $scfg, $volname, $snapname);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub volume_size_info
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $storeid, $volname, $timeout) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my ($vtype, $name, $vmid) = $class->parse_volname($volname);
|
|
||||||
my $info = _process_list($scfg, $storeid, run_cli($scfg, [ 'ls', $prefix.$name ]))->[0];
|
|
||||||
#return wantarray ? ($size, $format, $used, $parent, $st->ctime) : $size;
|
|
||||||
return $info->{size};
|
|
||||||
}
|
|
||||||
|
|
||||||
sub volume_resize
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $storeid, $volname, $size, $running) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my ($vtype, $name, $vmid) = $class->parse_volname($volname);
|
|
||||||
# $size is in bytes in this method
|
|
||||||
run_cli($scfg, [ 'modify', '--resize', (int(($size+4095)/4096)*4).'k', $prefix.$name ], json => 0);
|
|
||||||
return undef;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub volume_snapshot
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $storeid, $volname, $snap) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my ($vtype, $name, $vmid) = $class->parse_volname($volname);
|
|
||||||
run_cli($scfg, [ 'create', '--snapshot', $snap, $prefix.$name ], json => 0);
|
|
||||||
return undef;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub volume_snapshot_rollback
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $storeid, $volname, $snap) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my ($vtype, $name, $vmid) = $class->parse_volname($volname);
|
|
||||||
run_cli($scfg, [ 'rm', $prefix.$name ], json => 0);
|
|
||||||
run_cli($scfg, [ 'create', '--parent', $prefix.$name.'@'.$snap, $prefix.$name ], json => 0);
|
|
||||||
return undef;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub volume_snapshot_delete
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $storeid, $volname, $snap, $running) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my ($vtype, $name, $vmid) = $class->parse_volname($volname);
|
|
||||||
run_cli($scfg, [ 'rm', $prefix.$name.'@'.$snap ], json => 0);
|
|
||||||
return undef;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub volume_snapshot_needs_fsfreeze
|
|
||||||
{
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub volume_has_feature
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $feature, $storeid, $volname, $snapname, $running) = @_;
|
|
||||||
my $features = {
|
|
||||||
snapshot => { current => 1, snap => 1 },
|
|
||||||
clone => { base => 1, snap => 1 },
|
|
||||||
template => { current => 1 },
|
|
||||||
copy => { base => 1, current => 1, snap => 1 },
|
|
||||||
sparseinit => { base => 1, current => 1 },
|
|
||||||
rename => { current => 1 },
|
|
||||||
};
|
|
||||||
my ($vtype, $name, $vmid, $basename, $basevmid, $isBase) = $class->parse_volname($volname);
|
|
||||||
my $key = undef;
|
|
||||||
if ($snapname)
|
|
||||||
{
|
|
||||||
$key = 'snap';
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$key = $isBase ? 'base' : 'current';
|
|
||||||
}
|
|
||||||
return 1 if $features->{$feature}->{$key};
|
|
||||||
return undef;
|
|
||||||
}
|
|
||||||
|
|
||||||
sub rename_volume
|
|
||||||
{
|
|
||||||
my ($class, $scfg, $storeid, $source_volname, $target_vmid, $target_volname) = @_;
|
|
||||||
my $prefix = defined $scfg->{vitastor_prefix} ? $scfg->{vitastor_prefix} : 'pve/';
|
|
||||||
my (undef, $source_image, $source_vmid, $base_name, $base_vmid, undef, $format) =
|
|
||||||
$class->parse_volname($source_volname);
|
|
||||||
$target_volname = $class->find_free_diskname($storeid, $scfg, $target_vmid, $format) if !$target_volname;
|
|
||||||
run_cli($scfg, [ 'modify', '--rename', $prefix.$target_volname, $prefix.$source_image ], json => 0);
|
|
||||||
$base_name = $base_name ? "${base_name}/" : '';
|
|
||||||
return "${storeid}:${base_name}${target_volname}";
|
|
||||||
}
|
|
||||||
|
|
||||||
1;
|
|
|
@ -1,964 +0,0 @@
|
||||||
# Workaround:
|
|
||||||
# 461: allow to create volume from snapshot
|
|
||||||
# 527: adjust iodepth for faster deletion
|
|
||||||
# 591: add failure case when creating a volume
|
|
||||||
|
|
||||||
# Vitastor Driver for OpenStack Cinder
|
|
||||||
#
|
|
||||||
# --------------------------------------------
|
|
||||||
# Install as cinder/volume/drivers/vitastor.py
|
|
||||||
# --------------------------------------------
|
|
||||||
#
|
|
||||||
# Copyright 2020 Vitaliy Filippov
|
|
||||||
#
|
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
||||||
# not use this file except in compliance with the License. You may obtain
|
|
||||||
# a copy of the License at
|
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
# License for the specific language governing permissions and limitations
|
|
||||||
# under the License.
|
|
||||||
"""Cinder Vitastor Driver"""
|
|
||||||
|
|
||||||
import binascii
|
|
||||||
import base64
|
|
||||||
import errno
|
|
||||||
import json
|
|
||||||
import math
|
|
||||||
import os
|
|
||||||
import tempfile
|
|
||||||
|
|
||||||
from castellan import key_manager
|
|
||||||
from oslo_config import cfg
|
|
||||||
from oslo_log import log as logging
|
|
||||||
from oslo_service import loopingcall
|
|
||||||
from oslo_concurrency import processutils
|
|
||||||
from oslo_utils import encodeutils
|
|
||||||
from oslo_utils import excutils
|
|
||||||
from oslo_utils import fileutils
|
|
||||||
from oslo_utils import units
|
|
||||||
import six
|
|
||||||
from six.moves.urllib import request
|
|
||||||
|
|
||||||
from cinder import exception
|
|
||||||
from cinder.i18n import _
|
|
||||||
from cinder.image import image_utils
|
|
||||||
from cinder import interface
|
|
||||||
from cinder import objects
|
|
||||||
from cinder.objects import fields
|
|
||||||
from cinder import utils
|
|
||||||
from cinder.volume import configuration
|
|
||||||
from cinder.volume import driver
|
|
||||||
from cinder.volume import volume_utils
|
|
||||||
|
|
||||||
VERSION = '0.6.10'
|
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
|
||||||
|
|
||||||
VITASTOR_OPTS = [
|
|
||||||
cfg.StrOpt(
|
|
||||||
'vitastor_config_path',
|
|
||||||
default='/etc/vitastor/vitastor.conf',
|
|
||||||
help='Vitastor configuration file path'
|
|
||||||
),
|
|
||||||
cfg.StrOpt(
|
|
||||||
'vitastor_etcd_address',
|
|
||||||
default='',
|
|
||||||
help='Vitastor etcd address(es)'),
|
|
||||||
cfg.StrOpt(
|
|
||||||
'vitastor_etcd_prefix',
|
|
||||||
default='/vitastor',
|
|
||||||
help='Vitastor etcd prefix'
|
|
||||||
),
|
|
||||||
cfg.StrOpt(
|
|
||||||
'vitastor_pool_id',
|
|
||||||
default='',
|
|
||||||
help='Vitastor pool ID to use for volumes'
|
|
||||||
),
|
|
||||||
# FIXME exclusive_cinder_pool ?
|
|
||||||
]
|
|
||||||
|
|
||||||
CONF = cfg.CONF
|
|
||||||
CONF.register_opts(VITASTOR_OPTS, group = configuration.SHARED_CONF_GROUP)
|
|
||||||
|
|
||||||
class VitastorDriverException(exception.VolumeDriverException):
|
|
||||||
message = _("Vitastor Cinder driver failure: %(reason)s")
|
|
||||||
|
|
||||||
@interface.volumedriver
|
|
||||||
class VitastorDriver(driver.CloneableImageVD,
|
|
||||||
driver.ManageableVD, driver.ManageableSnapshotsVD,
|
|
||||||
driver.BaseVD):
|
|
||||||
"""Implements Vitastor volume commands."""
|
|
||||||
|
|
||||||
cfg = {}
|
|
||||||
_etcd_urls = []
|
|
||||||
|
|
||||||
def __init__(self, active_backend_id = None, *args, **kwargs):
|
|
||||||
super(VitastorDriver, self).__init__(*args, **kwargs)
|
|
||||||
self.configuration.append_config_values(VITASTOR_OPTS)
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def get_driver_options(cls):
|
|
||||||
additional_opts = cls._get_oslo_driver_opts(
|
|
||||||
'reserved_percentage',
|
|
||||||
'max_over_subscription_ratio',
|
|
||||||
'volume_dd_blocksize'
|
|
||||||
)
|
|
||||||
return VITASTOR_OPTS + additional_opts
|
|
||||||
|
|
||||||
def do_setup(self, context):
|
|
||||||
"""Performs initialization steps that could raise exceptions."""
|
|
||||||
super(VitastorDriver, self).do_setup(context)
|
|
||||||
# Make sure configuration is in UTF-8
|
|
||||||
for attr in [ 'config_path', 'etcd_address', 'etcd_prefix', 'pool_id' ]:
|
|
||||||
val = self.configuration.safe_get('vitastor_'+attr)
|
|
||||||
if val is not None:
|
|
||||||
self.cfg[attr] = utils.convert_str(val)
|
|
||||||
self.cfg = self._load_config(self.cfg)
|
|
||||||
|
|
||||||
def _load_config(self, cfg):
|
|
||||||
# Try to load configuration file
|
|
||||||
try:
|
|
||||||
f = open(cfg['config_path'] or '/etc/vitastor/vitastor.conf')
|
|
||||||
conf = json.loads(f.read())
|
|
||||||
f.close()
|
|
||||||
for k in conf:
|
|
||||||
cfg[k] = cfg.get(k, conf[k])
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
if isinstance(cfg['etcd_address'], str):
|
|
||||||
cfg['etcd_address'] = cfg['etcd_address'].split(',')
|
|
||||||
# Sanitize etcd URLs
|
|
||||||
for i, etcd_url in enumerate(cfg['etcd_address']):
|
|
||||||
ssl = False
|
|
||||||
if etcd_url.lower().startswith('http://'):
|
|
||||||
etcd_url = etcd_url[7:]
|
|
||||||
elif etcd_url.lower().startswith('https://'):
|
|
||||||
etcd_url = etcd_url[8:]
|
|
||||||
ssl = True
|
|
||||||
if etcd_url.find('/') < 0:
|
|
||||||
etcd_url += '/v3'
|
|
||||||
if ssl:
|
|
||||||
etcd_url = 'https://'+etcd_url
|
|
||||||
else:
|
|
||||||
etcd_url = 'http://'+etcd_url
|
|
||||||
cfg['etcd_address'][i] = etcd_url
|
|
||||||
return cfg
|
|
||||||
|
|
||||||
def check_for_setup_error(self):
|
|
||||||
"""Returns an error if prerequisites aren't met."""
|
|
||||||
|
|
||||||
def _encode_etcd_key(self, key):
|
|
||||||
if not isinstance(key, bytes):
|
|
||||||
key = str(key).encode('utf-8')
|
|
||||||
return base64.b64encode(self.cfg['etcd_prefix'].encode('utf-8')+b'/'+key).decode('utf-8')
|
|
||||||
|
|
||||||
def _encode_etcd_value(self, value):
|
|
||||||
if not isinstance(value, bytes):
|
|
||||||
value = str(value).encode('utf-8')
|
|
||||||
return base64.b64encode(value).decode('utf-8')
|
|
||||||
|
|
||||||
def _encode_etcd_requests(self, obj):
|
|
||||||
for v in obj:
|
|
||||||
for rt in v:
|
|
||||||
if 'key' in v[rt]:
|
|
||||||
v[rt]['key'] = self._encode_etcd_key(v[rt]['key'])
|
|
||||||
if 'range_end' in v[rt]:
|
|
||||||
v[rt]['range_end'] = self._encode_etcd_key(v[rt]['range_end'])
|
|
||||||
if 'value' in v[rt]:
|
|
||||||
v[rt]['value'] = self._encode_etcd_value(v[rt]['value'])
|
|
||||||
|
|
||||||
def _etcd_txn(self, params):
|
|
||||||
if 'compare' in params:
|
|
||||||
for v in params['compare']:
|
|
||||||
if 'key' in v:
|
|
||||||
v['key'] = self._encode_etcd_key(v['key'])
|
|
||||||
if 'failure' in params:
|
|
||||||
self._encode_etcd_requests(params['failure'])
|
|
||||||
if 'success' in params:
|
|
||||||
self._encode_etcd_requests(params['success'])
|
|
||||||
body = json.dumps(params).encode('utf-8')
|
|
||||||
headers = {
|
|
||||||
'Content-Type': 'application/json'
|
|
||||||
}
|
|
||||||
err = None
|
|
||||||
for etcd_url in self.cfg['etcd_address']:
|
|
||||||
try:
|
|
||||||
resp = request.urlopen(request.Request(etcd_url+'/kv/txn', body, headers), timeout = 5)
|
|
||||||
data = json.loads(resp.read())
|
|
||||||
if 'responses' not in data:
|
|
||||||
data['responses'] = []
|
|
||||||
for i, resp in enumerate(data['responses']):
|
|
||||||
if 'response_range' in resp:
|
|
||||||
if 'kvs' not in resp['response_range']:
|
|
||||||
resp['response_range']['kvs'] = []
|
|
||||||
for kv in resp['response_range']['kvs']:
|
|
||||||
kv['key'] = base64.b64decode(kv['key'].encode('utf-8')).decode('utf-8')
|
|
||||||
if kv['key'].startswith(self.cfg['etcd_prefix']+'/'):
|
|
||||||
kv['key'] = kv['key'][len(self.cfg['etcd_prefix'])+1 : ]
|
|
||||||
kv['value'] = json.loads(base64.b64decode(kv['value'].encode('utf-8')))
|
|
||||||
if len(resp.keys()) != 1:
|
|
||||||
LOG.exception('unknown responses['+str(i)+'] format: '+json.dumps(resp))
|
|
||||||
else:
|
|
||||||
resp = data['responses'][i] = resp[list(resp.keys())[0]]
|
|
||||||
return data
|
|
||||||
except Exception as e:
|
|
||||||
LOG.exception('error calling etcd transaction: '+body.decode('utf-8')+'\nerror: '+str(e))
|
|
||||||
err = e
|
|
||||||
raise err
|
|
||||||
|
|
||||||
def _etcd_foreach(self, prefix, add_fn):
|
|
||||||
total = 0
|
|
||||||
batch = 1000
|
|
||||||
begin = prefix+'/'
|
|
||||||
while True:
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': {
|
|
||||||
'key': begin,
|
|
||||||
'range_end': prefix+'0',
|
|
||||||
'limit': batch+1,
|
|
||||||
} },
|
|
||||||
] })
|
|
||||||
i = 0
|
|
||||||
while i < batch and i < len(resp['responses'][0]['kvs']):
|
|
||||||
kv = resp['responses'][0]['kvs'][i]
|
|
||||||
add_fn(kv)
|
|
||||||
i += 1
|
|
||||||
if len(resp['responses'][0]['kvs']) <= batch:
|
|
||||||
break
|
|
||||||
begin = resp['responses'][0]['kvs'][batch]['key']
|
|
||||||
return total
|
|
||||||
|
|
||||||
def _update_volume_stats(self):
|
|
||||||
location_info = json.dumps({
|
|
||||||
'config': self.configuration.vitastor_config_path,
|
|
||||||
'etcd_address': self.configuration.vitastor_etcd_address,
|
|
||||||
'etcd_prefix': self.configuration.vitastor_etcd_prefix,
|
|
||||||
'pool_id': self.configuration.vitastor_pool_id,
|
|
||||||
})
|
|
||||||
|
|
||||||
stats = {
|
|
||||||
'vendor_name': 'Vitastor',
|
|
||||||
'driver_version': self.VERSION,
|
|
||||||
'storage_protocol': 'vitastor',
|
|
||||||
'total_capacity_gb': 'unknown',
|
|
||||||
'free_capacity_gb': 'unknown',
|
|
||||||
# FIXME check if safe_get is required
|
|
||||||
'reserved_percentage': self.configuration.safe_get('reserved_percentage'),
|
|
||||||
'multiattach': True,
|
|
||||||
'thin_provisioning_support': True,
|
|
||||||
'max_over_subscription_ratio': self.configuration.safe_get('max_over_subscription_ratio'),
|
|
||||||
'location_info': location_info,
|
|
||||||
'backend_state': 'down',
|
|
||||||
'volume_backend_name': self.configuration.safe_get('volume_backend_name') or 'vitastor',
|
|
||||||
'replication_enabled': False,
|
|
||||||
}
|
|
||||||
|
|
||||||
try:
|
|
||||||
pool_stats = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'pool/stats/'+str(self.cfg['pool_id']) } }
|
|
||||||
] })
|
|
||||||
total_provisioned = 0
|
|
||||||
def add_total(kv):
|
|
||||||
nonlocal total_provisioned
|
|
||||||
if kv['key'].find('@') >= 0:
|
|
||||||
total_provisioned += kv['value']['size']
|
|
||||||
self._etcd_foreach('config/inode/'+str(self.cfg['pool_id']), lambda kv: add_total(kv))
|
|
||||||
stats['provisioned_capacity_gb'] = round(total_provisioned/1024.0/1024.0/1024.0, 2)
|
|
||||||
pool_stats = pool_stats['responses'][0]['kvs']
|
|
||||||
if len(pool_stats):
|
|
||||||
pool_stats = pool_stats[0]['value']
|
|
||||||
stats['free_capacity_gb'] = round(1024.0*(pool_stats['total_raw_tb']-pool_stats['used_raw_tb'])/pool_stats['raw_to_usable'], 2)
|
|
||||||
stats['total_capacity_gb'] = round(1024.0*pool_stats['total_raw_tb'], 2)
|
|
||||||
stats['backend_state'] = 'up'
|
|
||||||
except Exception as e:
|
|
||||||
# just log and return unknown capacities
|
|
||||||
LOG.exception('error getting vitastor pool stats: '+str(e))
|
|
||||||
|
|
||||||
self._stats = stats
|
|
||||||
|
|
||||||
def get_volume_stats(self, refresh=False):
|
|
||||||
"""Get volume stats.
|
|
||||||
If 'refresh' is True, run update the stats first.
|
|
||||||
"""
|
|
||||||
if not self._stats or refresh:
|
|
||||||
self._update_volume_stats()
|
|
||||||
|
|
||||||
return self._stats
|
|
||||||
|
|
||||||
def _next_id(self, resp):
|
|
||||||
if len(resp['kvs']) == 0:
|
|
||||||
return (1, 0)
|
|
||||||
else:
|
|
||||||
return (1 + resp['kvs'][0]['value'], resp['kvs'][0]['mod_revision'])
|
|
||||||
|
|
||||||
def create_volume(self, volume):
|
|
||||||
"""Creates a logical volume."""
|
|
||||||
|
|
||||||
size = int(volume.size) * units.Gi
|
|
||||||
# FIXME: Check if convert_str is really required
|
|
||||||
vol_name = utils.convert_str(volume.name)
|
|
||||||
if vol_name.find('@') >= 0 or vol_name.find('/') >= 0:
|
|
||||||
raise exception.VolumeBackendAPIException(data = '@ and / are forbidden in volume and snapshot names')
|
|
||||||
|
|
||||||
LOG.debug("creating volume '%s'", vol_name)
|
|
||||||
|
|
||||||
self._create_image(vol_name, { 'size': size })
|
|
||||||
|
|
||||||
if volume.encryption_key_id:
|
|
||||||
self._create_encrypted_volume(volume, volume.obj_context)
|
|
||||||
|
|
||||||
volume_update = {}
|
|
||||||
return volume_update
|
|
||||||
|
|
||||||
def _create_encrypted_volume(self, volume, context):
|
|
||||||
"""Create a new LUKS encrypted image directly in Vitastor."""
|
|
||||||
vol_name = utils.convert_str(volume.name)
|
|
||||||
f, opts = self._encrypt_opts(volume, context)
|
|
||||||
# FIXME: Check if it works at all :-)
|
|
||||||
self._execute(
|
|
||||||
'qemu-img', 'convert', '-f', 'luks', *opts,
|
|
||||||
'vitastor:image='+vol_name.replace(':', '\\:')+self._qemu_args(),
|
|
||||||
'%sM' % (volume.size * 1024)
|
|
||||||
)
|
|
||||||
f.close()
|
|
||||||
|
|
||||||
def _encrypt_opts(self, volume, context):
|
|
||||||
encryption = volume_utils.check_encryption_provider(self.db, volume, context)
|
|
||||||
# Fetch the key associated with the volume and decode the passphrase
|
|
||||||
keymgr = key_manager.API(CONF)
|
|
||||||
key = keymgr.get(context, encryption['encryption_key_id'])
|
|
||||||
passphrase = binascii.hexlify(key.get_encoded()).decode('utf-8')
|
|
||||||
# Decode the dm-crypt style cipher spec into something qemu-img can use
|
|
||||||
cipher_spec = image_utils.decode_cipher(encryption['cipher'], encryption['key_size'])
|
|
||||||
tmp_dir = volume_utils.image_conversion_dir()
|
|
||||||
f = tempfile.NamedTemporaryFile(prefix = 'luks_', dir = tmp_dir)
|
|
||||||
f.write(passphrase)
|
|
||||||
f.flush()
|
|
||||||
return (f, [
|
|
||||||
'--object', 'secret,id=luks_sec,format=raw,file=%(passfile)s' % {'passfile': f.name},
|
|
||||||
'-o', 'key-secret=luks_sec,cipher-alg=%(cipher_alg)s,cipher-mode=%(cipher_mode)s,ivgen-alg=%(ivgen_alg)s' % cipher_spec,
|
|
||||||
])
|
|
||||||
|
|
||||||
def create_snapshot(self, snapshot):
|
|
||||||
"""Creates a volume snapshot."""
|
|
||||||
|
|
||||||
vol_name = utils.convert_str(snapshot.volume_name)
|
|
||||||
snap_name = utils.convert_str(snapshot.name)
|
|
||||||
if snap_name.find('@') >= 0 or snap_name.find('/') >= 0:
|
|
||||||
raise exception.VolumeBackendAPIException(data = '@ and / are forbidden in volume and snapshot names')
|
|
||||||
self._create_snapshot(vol_name, vol_name+'@'+snap_name)
|
|
||||||
|
|
||||||
def snapshot_revert_use_temp_snapshot(self):
|
|
||||||
"""Disable the use of a temporary snapshot on revert."""
|
|
||||||
return False
|
|
||||||
|
|
||||||
def revert_to_snapshot(self, context, volume, snapshot):
|
|
||||||
"""Revert a volume to a given snapshot."""
|
|
||||||
|
|
||||||
# FIXME Delete the image, then recreate it from the snapshot
|
|
||||||
|
|
||||||
def delete_snapshot(self, snapshot):
|
|
||||||
"""Deletes a snapshot."""
|
|
||||||
|
|
||||||
vol_name = utils.convert_str(snapshot.volume_name)
|
|
||||||
snap_name = utils.convert_str(snapshot.name)
|
|
||||||
|
|
||||||
# Find the snapshot
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'index/image/'+vol_name+'@'+snap_name } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) == 0:
|
|
||||||
raise exception.SnapshotNotFound(snapshot_id = snap_name)
|
|
||||||
inode_id = int(resp['responses'][0]['kvs'][0]['value']['id'])
|
|
||||||
pool_id = int(resp['responses'][0]['kvs'][0]['value']['pool_id'])
|
|
||||||
parents = {}
|
|
||||||
parents[(pool_id << 48) | (inode_id & 0xffffffffffff)] = True
|
|
||||||
|
|
||||||
# Check if there are child volumes
|
|
||||||
children = self._child_count(parents)
|
|
||||||
if children > 0:
|
|
||||||
raise exception.SnapshotIsBusy(snapshot_name = snap_name)
|
|
||||||
|
|
||||||
# FIXME: We can't delete snapshots because we can't merge layers yet
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Snapshot delete (layer merge) is not implemented yet')
|
|
||||||
|
|
||||||
def _child_count(self, parents):
|
|
||||||
children = 0
|
|
||||||
def add_child(kv):
|
|
||||||
nonlocal children
|
|
||||||
children += self._check_parent(kv, parents)
|
|
||||||
self._etcd_foreach('config/inode', lambda kv: add_child(kv))
|
|
||||||
return children
|
|
||||||
|
|
||||||
def _check_parent(self, kv, parents):
|
|
||||||
if 'parent_id' not in kv['value']:
|
|
||||||
return 0
|
|
||||||
parent_id = kv['value']['parent_id']
|
|
||||||
_, _, pool_id, inode_id = kv['key'].split('/')
|
|
||||||
parent_pool_id = pool_id
|
|
||||||
if 'parent_pool_id' in kv['value'] and kv['value']['parent_pool_id']:
|
|
||||||
parent_pool_id = kv['value']['parent_pool_id']
|
|
||||||
inode = (int(pool_id) << 48) | (int(inode_id) & 0xffffffffffff)
|
|
||||||
parent = (int(parent_pool_id) << 48) | (int(parent_id) & 0xffffffffffff)
|
|
||||||
if parent in parents and inode not in parents:
|
|
||||||
return 1
|
|
||||||
return 0
|
|
||||||
|
|
||||||
def create_cloned_volume(self, volume, src_vref):
|
|
||||||
"""Create a cloned volume from another volume."""
|
|
||||||
|
|
||||||
size = int(volume.size) * units.Gi
|
|
||||||
src_name = utils.convert_str(src_vref.name)
|
|
||||||
dest_name = utils.convert_str(volume.name)
|
|
||||||
if dest_name.find('@') >= 0 or dest_name.find('/') >= 0:
|
|
||||||
raise exception.VolumeBackendAPIException(data = '@ and / are forbidden in volume and snapshot names')
|
|
||||||
|
|
||||||
# FIXME Do full copy if requested (cfg.disable_clone)
|
|
||||||
|
|
||||||
if src_vref.admin_metadata.get('readonly') == 'True':
|
|
||||||
# source volume is a volume-image cache entry or other readonly volume
|
|
||||||
# clone without intermediate snapshot
|
|
||||||
src = self._get_image(src_name)
|
|
||||||
LOG.debug("creating image '%s' from '%s'", dest_name, src_name)
|
|
||||||
new_cfg = self._create_image(dest_name, {
|
|
||||||
'size': size,
|
|
||||||
'parent_id': src['idx']['id'],
|
|
||||||
'parent_pool_id': src['idx']['pool_id'],
|
|
||||||
})
|
|
||||||
return {}
|
|
||||||
|
|
||||||
clone_snap = "%s@%s.clone_snap" % (src_name, dest_name)
|
|
||||||
make_img = True
|
|
||||||
if (volume.display_name and
|
|
||||||
volume.display_name.startswith('image-') and
|
|
||||||
src_vref.project_id != volume.project_id):
|
|
||||||
# idiotic openstack creates image-volume cache entries
|
|
||||||
# as clones of normal VM volumes... :-X prevent it :-D
|
|
||||||
clone_snap = dest_name
|
|
||||||
make_img = False
|
|
||||||
|
|
||||||
LOG.debug("creating layer '%s' under '%s'", clone_snap, src_name)
|
|
||||||
new_cfg = self._create_snapshot(src_name, clone_snap, True)
|
|
||||||
if make_img:
|
|
||||||
# Then create a clone from it
|
|
||||||
new_cfg = self._create_image(dest_name, {
|
|
||||||
'size': size,
|
|
||||||
'parent_id': new_cfg['parent_id'],
|
|
||||||
'parent_pool_id': new_cfg['parent_pool_id'],
|
|
||||||
})
|
|
||||||
|
|
||||||
return {}
|
|
||||||
|
|
||||||
def create_volume_from_snapshot(self, volume, snapshot):
|
|
||||||
"""Creates a cloned volume from an existing snapshot."""
|
|
||||||
|
|
||||||
vol_name = utils.convert_str(volume.name)
|
|
||||||
snap_name = utils.convert_str(snapshot.name)
|
|
||||||
|
|
||||||
snap = self._get_image("volume-"+snapshot.volume_id+'@'+snap_name)
|
|
||||||
if not snap:
|
|
||||||
raise exception.SnapshotNotFound(snapshot_id = snap_name)
|
|
||||||
snap_inode_id = int(resp['responses'][0]['kvs'][0]['value']['id'])
|
|
||||||
snap_pool_id = int(resp['responses'][0]['kvs'][0]['value']['pool_id'])
|
|
||||||
|
|
||||||
size = snap['cfg']['size']
|
|
||||||
if int(volume.size):
|
|
||||||
size = int(volume.size) * units.Gi
|
|
||||||
new_cfg = self._create_image(vol_name, {
|
|
||||||
'size': size,
|
|
||||||
'parent_id': snap['idx']['id'],
|
|
||||||
'parent_pool_id': snap['idx']['pool_id'],
|
|
||||||
})
|
|
||||||
|
|
||||||
return {}
|
|
||||||
|
|
||||||
def _vitastor_args(self):
|
|
||||||
args = []
|
|
||||||
for k in [ 'config_path', 'etcd_address', 'etcd_prefix' ]:
|
|
||||||
v = self.configuration.safe_get('vitastor_'+k)
|
|
||||||
if v:
|
|
||||||
args.extend(['--'+k, v])
|
|
||||||
return args
|
|
||||||
|
|
||||||
def _qemu_args(self):
|
|
||||||
args = ''
|
|
||||||
for k in [ 'config_path', 'etcd_address', 'etcd_prefix' ]:
|
|
||||||
v = self.configuration.safe_get('vitastor_'+k)
|
|
||||||
kk = k
|
|
||||||
if kk == 'etcd_address':
|
|
||||||
# FIXME use etcd_address in qemu driver
|
|
||||||
kk = 'etcd_host'
|
|
||||||
if v:
|
|
||||||
args += ':'+kk.replace('_', '-')+'='+v.replace(':', '\\:')
|
|
||||||
return args
|
|
||||||
|
|
||||||
def delete_volume(self, volume):
|
|
||||||
"""Deletes a logical volume."""
|
|
||||||
|
|
||||||
vol_name = utils.convert_str(volume.name)
|
|
||||||
|
|
||||||
# Find the volume and all its snapshots
|
|
||||||
range_end = b'index/image/' + vol_name.encode('utf-8')
|
|
||||||
range_end = range_end[0 : len(range_end)-1] + six.int2byte(range_end[len(range_end)-1] + 1)
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'index/image/'+vol_name, 'range_end': range_end } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) == 0:
|
|
||||||
# already deleted
|
|
||||||
LOG.info("volume %s no longer exists in backend", vol_name)
|
|
||||||
return
|
|
||||||
layers = resp['responses'][0]['kvs']
|
|
||||||
layer_ids = {}
|
|
||||||
for kv in layers:
|
|
||||||
inode_id = int(kv['value']['id'])
|
|
||||||
pool_id = int(kv['value']['pool_id'])
|
|
||||||
inode_pool_id = (pool_id << 48) | (inode_id & 0xffffffffffff)
|
|
||||||
layer_ids[inode_pool_id] = True
|
|
||||||
|
|
||||||
# Check if the volume has clones and raise 'busy' if so
|
|
||||||
children = self._child_count(layer_ids)
|
|
||||||
if children > 0:
|
|
||||||
raise exception.VolumeIsBusy(volume_name = vol_name)
|
|
||||||
|
|
||||||
# Clear data
|
|
||||||
for kv in layers:
|
|
||||||
args = [
|
|
||||||
'vitastor-cli', 'rm-data', '--pool', str(kv['value']['pool_id']),
|
|
||||||
'--inode', str(kv['value']['id']), '--iodepth', '4', '--progress', '0',
|
|
||||||
*(self._vitastor_args())
|
|
||||||
]
|
|
||||||
try:
|
|
||||||
self._execute(*args)
|
|
||||||
except processutils.ProcessExecutionError as exc:
|
|
||||||
LOG.error("Failed to remove layer "+kv['key']+": "+exc)
|
|
||||||
raise exception.VolumeBackendAPIException(data = exc.stderr)
|
|
||||||
|
|
||||||
# Delete all layers from etcd
|
|
||||||
requests = []
|
|
||||||
for kv in layers:
|
|
||||||
requests.append({ 'request_delete_range': { 'key': kv['key'] } })
|
|
||||||
requests.append({ 'request_delete_range': { 'key': 'config/inode/'+str(kv['value']['pool_id'])+'/'+str(kv['value']['id']) } })
|
|
||||||
self._etcd_txn({ 'success': requests })
|
|
||||||
|
|
||||||
def retype(self, context, volume, new_type, diff, host):
|
|
||||||
"""Change extra type specifications for a volume."""
|
|
||||||
|
|
||||||
# FIXME Maybe (in the future) support multiple pools as different types
|
|
||||||
return True, {}
|
|
||||||
|
|
||||||
def ensure_export(self, context, volume):
|
|
||||||
"""Synchronously recreates an export for a logical volume."""
|
|
||||||
pass
|
|
||||||
|
|
||||||
def create_export(self, context, volume, connector):
|
|
||||||
"""Exports the volume."""
|
|
||||||
pass
|
|
||||||
|
|
||||||
def remove_export(self, context, volume):
|
|
||||||
"""Removes an export for a logical volume."""
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _create_image(self, vol_name, cfg):
|
|
||||||
pool_s = str(self.cfg['pool_id'])
|
|
||||||
image_id = 0
|
|
||||||
while image_id == 0:
|
|
||||||
# check if the image already exists and find a free ID
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'index/image/'+vol_name } },
|
|
||||||
{ 'request_range': { 'key': 'index/maxid/'+pool_s } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) > 0:
|
|
||||||
# already exists
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Volume '+vol_name+' already exists')
|
|
||||||
image_id, id_mod = self._next_id(resp['responses'][1])
|
|
||||||
# try to create the image
|
|
||||||
resp = self._etcd_txn({ 'compare': [
|
|
||||||
{ 'target': 'MOD', 'mod_revision': id_mod, 'key': 'index/maxid/'+pool_s },
|
|
||||||
{ 'target': 'VERSION', 'version': 0, 'key': 'index/image/'+vol_name },
|
|
||||||
{ 'target': 'VERSION', 'version': 0, 'key': 'config/inode/'+pool_s+'/'+str(image_id) },
|
|
||||||
], 'success': [
|
|
||||||
{ 'request_put': { 'key': 'index/maxid/'+pool_s, 'value': image_id } },
|
|
||||||
{ 'request_put': { 'key': 'index/image/'+vol_name, 'value': json.dumps({
|
|
||||||
'id': image_id, 'pool_id': self.cfg['pool_id']
|
|
||||||
}) } },
|
|
||||||
{ 'request_put': { 'key': 'config/inode/'+pool_s+'/'+str(image_id), 'value': json.dumps({
|
|
||||||
**cfg, 'name': vol_name,
|
|
||||||
}) } },
|
|
||||||
], 'failure': [
|
|
||||||
{ 'request_put': { 'key': 'index/maxid/'+pool_s, 'value': image_id } },
|
|
||||||
] })
|
|
||||||
if not resp.get('succeeded'):
|
|
||||||
# repeat
|
|
||||||
image_id = 0
|
|
||||||
|
|
||||||
def _create_snapshot(self, vol_name, snap_vol_name, allow_existing = False):
|
|
||||||
while True:
|
|
||||||
# check if the image already exists and snapshot doesn't
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'index/image/'+vol_name } },
|
|
||||||
{ 'request_range': { 'key': 'index/image/'+snap_vol_name } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) == 0:
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Volume '+vol_name+' does not exist')
|
|
||||||
if len(resp['responses'][1]['kvs']) > 0:
|
|
||||||
if allow_existing:
|
|
||||||
snap_idx = resp['responses'][1]['kvs'][0]['value']
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'config/inode/'+str(snap_idx['pool_id'])+'/'+str(snap_idx['id']) } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) == 0:
|
|
||||||
raise exception.VolumeBackendAPIException(data =
|
|
||||||
'Volume '+snap_vol_name+' is already indexed, but does not exist'
|
|
||||||
)
|
|
||||||
return resp['responses'][0]['kvs'][0]['value']
|
|
||||||
raise exception.VolumeBackendAPIException(
|
|
||||||
data = 'Volume '+snap_vol_name+' already exists'
|
|
||||||
)
|
|
||||||
vol_idx = resp['responses'][0]['kvs'][0]['value']
|
|
||||||
vol_idx_mod = resp['responses'][0]['kvs'][0]['mod_revision']
|
|
||||||
# get image inode config and find a new ID
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'config/inode/'+str(vol_idx['pool_id'])+'/'+str(vol_idx['id']) } },
|
|
||||||
{ 'request_range': { 'key': 'index/maxid/'+str(self.cfg['pool_id']) } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) == 0:
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Volume '+vol_name+' does not exist')
|
|
||||||
vol_cfg = resp['responses'][0]['kvs'][0]['value']
|
|
||||||
vol_mod = resp['responses'][0]['kvs'][0]['mod_revision']
|
|
||||||
new_id, id_mod = self._next_id(resp['responses'][1])
|
|
||||||
# try to redirect image to the new inode
|
|
||||||
new_cfg = {
|
|
||||||
**vol_cfg, 'name': vol_name, 'parent_id': vol_idx['id'], 'parent_pool_id': vol_idx['pool_id']
|
|
||||||
}
|
|
||||||
resp = self._etcd_txn({ 'compare': [
|
|
||||||
{ 'target': 'MOD', 'mod_revision': vol_idx_mod, 'key': 'index/image/'+vol_name },
|
|
||||||
{ 'target': 'MOD', 'mod_revision': vol_mod, 'key': 'config/inode/'+str(vol_idx['pool_id'])+'/'+str(vol_idx['id']) },
|
|
||||||
{ 'target': 'MOD', 'mod_revision': id_mod, 'key': 'index/maxid/'+str(self.cfg['pool_id']) },
|
|
||||||
{ 'target': 'VERSION', 'version': 0, 'key': 'index/image/'+snap_vol_name },
|
|
||||||
{ 'target': 'VERSION', 'version': 0, 'key': 'config/inode/'+str(self.cfg['pool_id'])+'/'+str(new_id) },
|
|
||||||
], 'success': [
|
|
||||||
{ 'request_put': { 'key': 'index/maxid/'+str(self.cfg['pool_id']), 'value': new_id } },
|
|
||||||
{ 'request_put': { 'key': 'index/image/'+vol_name, 'value': json.dumps({
|
|
||||||
'id': new_id, 'pool_id': self.cfg['pool_id']
|
|
||||||
}) } },
|
|
||||||
{ 'request_put': { 'key': 'config/inode/'+str(self.cfg['pool_id'])+'/'+str(new_id), 'value': json.dumps(new_cfg) } },
|
|
||||||
{ 'request_put': { 'key': 'index/image/'+snap_vol_name, 'value': json.dumps({
|
|
||||||
'id': vol_idx['id'], 'pool_id': vol_idx['pool_id']
|
|
||||||
}) } },
|
|
||||||
{ 'request_put': { 'key': 'config/inode/'+str(vol_idx['pool_id'])+'/'+str(vol_idx['id']), 'value': json.dumps({
|
|
||||||
**vol_cfg, 'name': snap_vol_name, 'readonly': True
|
|
||||||
}) } }
|
|
||||||
] })
|
|
||||||
if resp.get('succeeded'):
|
|
||||||
return new_cfg
|
|
||||||
|
|
||||||
def initialize_connection(self, volume, connector):
|
|
||||||
data = {
|
|
||||||
'driver_volume_type': 'vitastor',
|
|
||||||
'data': {
|
|
||||||
'config_path': self.configuration.vitastor_config_path,
|
|
||||||
'etcd_address': self.configuration.vitastor_etcd_address,
|
|
||||||
'etcd_prefix': self.configuration.vitastor_etcd_prefix,
|
|
||||||
'name': volume.name,
|
|
||||||
'logical_block_size': '512',
|
|
||||||
'physical_block_size': '4096',
|
|
||||||
}
|
|
||||||
}
|
|
||||||
LOG.debug('connection data: %s', data)
|
|
||||||
return data
|
|
||||||
|
|
||||||
def terminate_connection(self, volume, connector, **kwargs):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def clone_image(self, context, volume, image_location, image_meta, image_service):
|
|
||||||
if image_location:
|
|
||||||
# Note: image_location[0] is glance image direct_url.
|
|
||||||
# image_location[1] contains the list of all locations (including
|
|
||||||
# direct_url) or None if show_multiple_locations is False in
|
|
||||||
# glance configuration.
|
|
||||||
if image_location[1]:
|
|
||||||
url_locations = [location['url'] for location in image_location[1]]
|
|
||||||
else:
|
|
||||||
url_locations = [image_location[0]]
|
|
||||||
# iterate all locations to look for a cloneable one.
|
|
||||||
for url_location in url_locations:
|
|
||||||
if url_location and url_location.startswith('cinder://'):
|
|
||||||
# The idea is to use cinder://<volume-id> Glance volumes as base images
|
|
||||||
base_vol = self.db.volume_get(context, url_location[len('cinder://') : ])
|
|
||||||
if not base_vol or base_vol.volume_type_id != volume.volume_type_id:
|
|
||||||
continue
|
|
||||||
size = int(volume.size) * units.Gi
|
|
||||||
dest_name = utils.convert_str(volume.name)
|
|
||||||
# Find or create the base snapshot
|
|
||||||
snap_cfg = self._create_snapshot(base_vol.name, base_vol.name+'@.clone_snap', True)
|
|
||||||
# Then create a clone from it
|
|
||||||
new_cfg = self._create_image(dest_name, {
|
|
||||||
'size': size,
|
|
||||||
'parent_id': snap_cfg['parent_id'],
|
|
||||||
'parent_pool_id': snap_cfg['parent_pool_id'],
|
|
||||||
})
|
|
||||||
return ({}, True)
|
|
||||||
return ({}, False)
|
|
||||||
|
|
||||||
def copy_image_to_encrypted_volume(self, context, volume, image_service, image_id):
|
|
||||||
self.copy_image_to_volume(context, volume, image_service, image_id, encrypted = True)
|
|
||||||
|
|
||||||
def copy_image_to_volume(self, context, volume, image_service, image_id, encrypted = False):
|
|
||||||
tmp_dir = volume_utils.image_conversion_dir()
|
|
||||||
with tempfile.NamedTemporaryFile(dir = tmp_dir) as tmp:
|
|
||||||
image_utils.fetch_to_raw(
|
|
||||||
context, image_service, image_id, tmp.name,
|
|
||||||
self.configuration.volume_dd_blocksize, size = volume.size
|
|
||||||
)
|
|
||||||
out_format = [ '-O', 'raw' ]
|
|
||||||
if encrypted:
|
|
||||||
key_file, opts = self._encrypt_opts(volume, context)
|
|
||||||
out_format = [ '-O', 'luks', *opts ]
|
|
||||||
dest_name = utils.convert_str(volume.name)
|
|
||||||
self._try_execute(
|
|
||||||
'qemu-img', 'convert', '-f', 'raw', tmp.name, *out_format,
|
|
||||||
'vitastor:image='+dest_name.replace(':', '\\:')+self._qemu_args()
|
|
||||||
)
|
|
||||||
if encrypted:
|
|
||||||
key_file.close()
|
|
||||||
|
|
||||||
def copy_volume_to_image(self, context, volume, image_service, image_meta):
|
|
||||||
tmp_dir = volume_utils.image_conversion_dir()
|
|
||||||
tmp_file = os.path.join(tmp_dir, volume.name + '-' + image_meta['id'])
|
|
||||||
with fileutils.remove_path_on_error(tmp_file):
|
|
||||||
vol_name = utils.convert_str(volume.name)
|
|
||||||
self._try_execute(
|
|
||||||
'qemu-img', 'convert', '-f', 'raw',
|
|
||||||
'vitastor:image='+vol_name.replace(':', '\\:')+self._qemu_args(),
|
|
||||||
'-O', 'raw', tmp_file
|
|
||||||
)
|
|
||||||
# FIXME: Copy directly if the destination image is also in Vitastor
|
|
||||||
volume_utils.upload_volume(context, image_service, image_meta, tmp_file, volume)
|
|
||||||
os.unlink(tmp_file)
|
|
||||||
|
|
||||||
def _get_image(self, vol_name):
|
|
||||||
# find the image
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'index/image/'+vol_name } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) == 0:
|
|
||||||
return None
|
|
||||||
vol_idx = resp['responses'][0]['kvs'][0]['value']
|
|
||||||
vol_idx_mod = resp['responses'][0]['kvs'][0]['mod_revision']
|
|
||||||
# get image inode config
|
|
||||||
resp = self._etcd_txn({ 'success': [
|
|
||||||
{ 'request_range': { 'key': 'config/inode/'+str(vol_idx['pool_id'])+'/'+str(vol_idx['id']) } },
|
|
||||||
] })
|
|
||||||
if len(resp['responses'][0]['kvs']) == 0:
|
|
||||||
return None
|
|
||||||
vol_cfg = resp['responses'][0]['kvs'][0]['value']
|
|
||||||
vol_cfg_mod = resp['responses'][0]['kvs'][0]['mod_revision']
|
|
||||||
return {
|
|
||||||
'cfg': vol_cfg,
|
|
||||||
'cfg_mod': vol_cfg_mod,
|
|
||||||
'idx': vol_idx,
|
|
||||||
'idx_mod': vol_idx_mod,
|
|
||||||
}
|
|
||||||
|
|
||||||
def extend_volume(self, volume, new_size):
|
|
||||||
"""Extend an existing volume."""
|
|
||||||
vol_name = utils.convert_str(volume.name)
|
|
||||||
while True:
|
|
||||||
vol = self._get_image(vol_name)
|
|
||||||
if not vol:
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Volume '+vol_name+' does not exist')
|
|
||||||
# change size
|
|
||||||
size = int(new_size) * units.Gi
|
|
||||||
if size == vol['cfg']['size']:
|
|
||||||
break
|
|
||||||
resp = self._etcd_txn({ 'compare': [ {
|
|
||||||
'target': 'MOD',
|
|
||||||
'mod_revision': vol['cfg_mod'],
|
|
||||||
'key': 'config/inode/'+str(vol['idx']['pool_id'])+'/'+str(vol['idx']['id']),
|
|
||||||
} ], 'success': [
|
|
||||||
{ 'request_put': {
|
|
||||||
'key': 'config/inode/'+str(vol['idx']['pool_id'])+'/'+str(vol['idx']['id']),
|
|
||||||
'value': json.dumps({ **vol['cfg'], 'size': size }),
|
|
||||||
} },
|
|
||||||
] })
|
|
||||||
if resp.get('succeeded'):
|
|
||||||
break
|
|
||||||
LOG.debug(
|
|
||||||
"Extend volume from %(old_size)s GB to %(new_size)s GB.",
|
|
||||||
{'old_size': volume.size, 'new_size': new_size}
|
|
||||||
)
|
|
||||||
|
|
||||||
def _add_manageable_volume(self, kv, manageable_volumes, cinder_ids):
|
|
||||||
cfg = kv['value']
|
|
||||||
if kv['key'].find('@') >= 0:
|
|
||||||
# snapshot
|
|
||||||
return
|
|
||||||
image_id = volume_utils.extract_id_from_volume_name(cfg['name'])
|
|
||||||
image_info = {
|
|
||||||
'reference': {'source-name': image_name},
|
|
||||||
'size': int(math.ceil(float(cfg['size']) / units.Gi)),
|
|
||||||
'cinder_id': None,
|
|
||||||
'extra_info': None,
|
|
||||||
}
|
|
||||||
if image_id in cinder_ids:
|
|
||||||
image_info['cinder_id'] = image_id
|
|
||||||
image_info['safe_to_manage'] = False
|
|
||||||
image_info['reason_not_safe'] = 'already managed'
|
|
||||||
else:
|
|
||||||
image_info['safe_to_manage'] = True
|
|
||||||
image_info['reason_not_safe'] = None
|
|
||||||
manageable_volumes.append(image_info)
|
|
||||||
|
|
||||||
def get_manageable_volumes(self, cinder_volumes, marker, limit, offset, sort_keys, sort_dirs):
|
|
||||||
manageable_volumes = []
|
|
||||||
cinder_ids = [resource['id'] for resource in cinder_volumes]
|
|
||||||
|
|
||||||
# List all volumes
|
|
||||||
# FIXME: It's possible to use pagination in our case, but.. do we want it?
|
|
||||||
self._etcd_foreach('config/inode/'+str(self.cfg['pool_id']),
|
|
||||||
lambda kv: self._add_manageable_volume(kv, manageable_volumes, cinder_ids))
|
|
||||||
|
|
||||||
return volume_utils.paginate_entries_list(
|
|
||||||
manageable_volumes, marker, limit, offset, sort_keys, sort_dirs)
|
|
||||||
|
|
||||||
def _get_existing_name(existing_ref):
|
|
||||||
if not isinstance(existing_ref, dict):
|
|
||||||
existing_ref = {"source-name": existing_ref}
|
|
||||||
if 'source-name' not in existing_ref:
|
|
||||||
reason = _('Reference must contain source-name element.')
|
|
||||||
raise exception.ManageExistingInvalidReference(existing_ref=existing_ref, reason=reason)
|
|
||||||
src_name = utils.convert_str(existing_ref['source-name'])
|
|
||||||
if not src_name:
|
|
||||||
reason = _('Reference must contain source-name element.')
|
|
||||||
raise exception.ManageExistingInvalidReference(existing_ref=existing_ref, reason=reason)
|
|
||||||
return src_name
|
|
||||||
|
|
||||||
def manage_existing_get_size(self, volume, existing_ref):
|
|
||||||
"""Return size of an existing image for manage_existing.
|
|
||||||
|
|
||||||
:param volume: volume ref info to be set
|
|
||||||
:param existing_ref: {'source-name': <image name>}
|
|
||||||
"""
|
|
||||||
src_name = self._get_existing_name(existing_ref)
|
|
||||||
vol = self._get_image(src_name)
|
|
||||||
if not vol:
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Volume '+src_name+' does not exist')
|
|
||||||
return int(math.ceil(float(vol['cfg']['size']) / units.Gi))
|
|
||||||
|
|
||||||
def manage_existing(self, volume, existing_ref):
|
|
||||||
"""Manages an existing image.
|
|
||||||
|
|
||||||
Renames the image name to match the expected name for the volume.
|
|
||||||
|
|
||||||
:param volume: volume ref info to be set
|
|
||||||
:param existing_ref: {'source-name': <image name>}
|
|
||||||
"""
|
|
||||||
from_name = self._get_existing_name(existing_ref)
|
|
||||||
to_name = utils.convert_str(volume.name)
|
|
||||||
self._rename(from_name, to_name)
|
|
||||||
|
|
||||||
def _rename(self, from_name, to_name):
|
|
||||||
while True:
|
|
||||||
vol = self._get_image(from_name)
|
|
||||||
if not vol:
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Volume '+from_name+' does not exist')
|
|
||||||
to = self._get_image(to_name)
|
|
||||||
if to:
|
|
||||||
raise exception.VolumeBackendAPIException(data = 'Volume '+to_name+' already exists')
|
|
||||||
resp = self._etcd_txn({ 'compare': [
|
|
||||||
{ 'target': 'MOD', 'mod_revision': vol['idx_mod'], 'key': 'index/image/'+vol['cfg']['name'] },
|
|
||||||
{ 'target': 'MOD', 'mod_revision': vol['cfg_mod'], 'key': 'config/inode/'+str(vol['idx']['pool_id'])+'/'+str(vol['idx']['id']) },
|
|
||||||
{ 'target': 'VERSION', 'version': 0, 'key': 'index/image/'+to_name },
|
|
||||||
], 'success': [
|
|
||||||
{ 'request_delete_range': { 'key': 'index/image/'+vol['cfg']['name'] } },
|
|
||||||
{ 'request_put': { 'key': 'index/image/'+to_name, 'value': json.dumps(vol['idx']) } },
|
|
||||||
{ 'request_put': { 'key': 'config/inode/'+str(vol['idx']['pool_id'])+'/'+str(vol['idx']['id']),
|
|
||||||
'value': json.dumps({ **vol['cfg'], 'name': to_name }) } },
|
|
||||||
] })
|
|
||||||
if resp.get('succeeded'):
|
|
||||||
break
|
|
||||||
|
|
||||||
def unmanage(self, volume):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _add_manageable_snapshot(self, kv, manageable_snapshots, cinder_ids):
|
|
||||||
cfg = kv['value']
|
|
||||||
dog = kv['key'].find('@')
|
|
||||||
if dog < 0:
|
|
||||||
# snapshot
|
|
||||||
return
|
|
||||||
image_name = kv['key'][0 : dog]
|
|
||||||
snap_name = kv['key'][dog+1 : ]
|
|
||||||
snapshot_id = volume_utils.extract_id_from_snapshot_name(snap_name)
|
|
||||||
snapshot_info = {
|
|
||||||
'reference': {'source-name': snap_name},
|
|
||||||
'size': int(math.ceil(float(cfg['size']) / units.Gi)),
|
|
||||||
'cinder_id': None,
|
|
||||||
'extra_info': None,
|
|
||||||
'safe_to_manage': False,
|
|
||||||
'reason_not_safe': None,
|
|
||||||
'source_reference': {'source-name': image_name}
|
|
||||||
}
|
|
||||||
if snapshot_id in cinder_ids:
|
|
||||||
# Exclude snapshots already managed.
|
|
||||||
snapshot_info['reason_not_safe'] = ('already managed')
|
|
||||||
snapshot_info['cinder_id'] = snapshot_id
|
|
||||||
elif snap_name.endswith('.clone_snap'):
|
|
||||||
# Exclude clone snapshot.
|
|
||||||
snapshot_info['reason_not_safe'] = ('used for clone snap')
|
|
||||||
else:
|
|
||||||
snapshot_info['safe_to_manage'] = True
|
|
||||||
manageable_snapshots.append(snapshot_info)
|
|
||||||
|
|
||||||
def get_manageable_snapshots(self, cinder_snapshots, marker, limit, offset, sort_keys, sort_dirs):
|
|
||||||
"""List manageable snapshots in Vitastor."""
|
|
||||||
manageable_snapshots = []
|
|
||||||
cinder_snapshot_ids = [resource['id'] for resource in cinder_snapshots]
|
|
||||||
# List all volumes
|
|
||||||
# FIXME: It's possible to use pagination in our case, but.. do we want it?
|
|
||||||
self._etcd_foreach('config/inode/'+str(self.cfg['pool_id']),
|
|
||||||
lambda kv: self._add_manageable_volume(kv, manageable_snapshots, cinder_snapshot_ids))
|
|
||||||
return volume_utils.paginate_entries_list(
|
|
||||||
manageable_snapshots, marker, limit, offset, sort_keys, sort_dirs)
|
|
||||||
|
|
||||||
def manage_existing_snapshot_get_size(self, snapshot, existing_ref):
|
|
||||||
"""Return size of an existing image for manage_existing.
|
|
||||||
|
|
||||||
:param snapshot: snapshot ref info to be set
|
|
||||||
:param existing_ref: {'source-name': <name of snapshot>}
|
|
||||||
"""
|
|
||||||
vol_name = utils.convert_str(snapshot.volume_name)
|
|
||||||
snap_name = self._get_existing_name(existing_ref)
|
|
||||||
vol = self._get_image(vol_name+'@'+snap_name)
|
|
||||||
if not vol:
|
|
||||||
raise exception.ManageExistingInvalidReference(
|
|
||||||
existing_ref=snapshot_name, reason='Specified snapshot does not exist.'
|
|
||||||
)
|
|
||||||
return int(math.ceil(float(vol['cfg']['size']) / units.Gi))
|
|
||||||
|
|
||||||
def manage_existing_snapshot(self, snapshot, existing_ref):
|
|
||||||
"""Manages an existing snapshot.
|
|
||||||
|
|
||||||
Renames the snapshot name to match the expected name for the snapshot.
|
|
||||||
Error checking done by manage_existing_get_size is not repeated.
|
|
||||||
|
|
||||||
:param snapshot: snapshot ref info to be set
|
|
||||||
:param existing_ref: {'source-name': <name of snapshot>}
|
|
||||||
"""
|
|
||||||
vol_name = utils.convert_str(snapshot.volume_name)
|
|
||||||
snap_name = self._get_existing_name(existing_ref)
|
|
||||||
from_name = vol_name+'@'+snap_name
|
|
||||||
to_name = vol_name+'@'+utils.convert_str(snapshot.name)
|
|
||||||
self._rename(from_name, to_name)
|
|
||||||
|
|
||||||
def unmanage_snapshot(self, snapshot):
|
|
||||||
"""Removes the specified snapshot from Cinder management."""
|
|
||||||
pass
|
|
||||||
|
|
||||||
def _dumps(self, obj):
|
|
||||||
return json.dumps(obj, separators=(',', ':'), sort_keys=True)
|
|
|
@ -1,23 +0,0 @@
|
||||||
# Devstack configuration for bridged networking
|
|
||||||
|
|
||||||
[[local|localrc]]
|
|
||||||
ADMIN_PASSWORD=secret
|
|
||||||
DATABASE_PASSWORD=$ADMIN_PASSWORD
|
|
||||||
RABBIT_PASSWORD=$ADMIN_PASSWORD
|
|
||||||
SERVICE_PASSWORD=$ADMIN_PASSWORD
|
|
||||||
HOST_IP=10.0.2.15
|
|
||||||
Q_USE_SECGROUP=True
|
|
||||||
FLOATING_RANGE="10.0.2.0/24"
|
|
||||||
IPV4_ADDRS_SAFE_TO_USE="10.0.5.0/24"
|
|
||||||
Q_FLOATING_ALLOCATION_POOL=start=10.0.2.50,end=10.0.2.100
|
|
||||||
PUBLIC_NETWORK_GATEWAY=10.0.2.2
|
|
||||||
PUBLIC_INTERFACE=ens3
|
|
||||||
Q_USE_PROVIDERNET_FOR_PUBLIC=True
|
|
||||||
Q_AGENT=linuxbridge
|
|
||||||
Q_ML2_PLUGIN_MECHANISM_DRIVERS=linuxbridge
|
|
||||||
LB_PHYSICAL_INTERFACE=ens3
|
|
||||||
PUBLIC_PHYSICAL_NETWORK=default
|
|
||||||
LB_INTERFACE_MAPPINGS=default:ens3
|
|
||||||
Q_SERVICE_PLUGIN_CLASSES=
|
|
||||||
Q_ML2_PLUGIN_TYPE_DRIVERS=flat
|
|
||||||
Q_ML2_PLUGIN_EXT_DRIVERS=
|
|
|
@ -1,637 +0,0 @@
|
||||||
commit 7f01510ef207940b07fac4f5fc8b9f1580b443aa
|
|
||||||
Author: Vitaliy Filippov <vitalif@yourcmc.ru>
|
|
||||||
Date: Sun Jun 27 12:52:40 2021 +0300
|
|
||||||
|
|
||||||
Add Vitastor support
|
|
||||||
|
|
||||||
diff --git a/docs/schemas/domaincommon.rng b/docs/schemas/domaincommon.rng
|
|
||||||
index aa50eac..082b4f8 100644
|
|
||||||
--- a/docs/schemas/domaincommon.rng
|
|
||||||
+++ b/docs/schemas/domaincommon.rng
|
|
||||||
@@ -1728,6 +1728,35 @@
|
|
||||||
</element>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
+ <define name="diskSourceNetworkProtocolVitastor">
|
|
||||||
+ <element name="source">
|
|
||||||
+ <interleave>
|
|
||||||
+ <attribute name="protocol">
|
|
||||||
+ <value>vitastor</value>
|
|
||||||
+ </attribute>
|
|
||||||
+ <ref name="diskSourceCommon"/>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="name"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="query"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <zeroOrMore>
|
|
||||||
+ <ref name="diskSourceNetworkHost"/>
|
|
||||||
+ </zeroOrMore>
|
|
||||||
+ <optional>
|
|
||||||
+ <element name="config">
|
|
||||||
+ <attribute name="file">
|
|
||||||
+ <ref name="absFilePath"/>
|
|
||||||
+ </attribute>
|
|
||||||
+ <empty/>
|
|
||||||
+ </element>
|
|
||||||
+ </optional>
|
|
||||||
+ <empty/>
|
|
||||||
+ </interleave>
|
|
||||||
+ </element>
|
|
||||||
+ </define>
|
|
||||||
+
|
|
||||||
<define name="diskSourceNetworkProtocolISCSI">
|
|
||||||
<element name="source">
|
|
||||||
<attribute name="protocol">
|
|
||||||
@@ -1851,6 +1880,7 @@
|
|
||||||
<ref name="diskSourceNetworkProtocolHTTP"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolSimple"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolVxHS"/>
|
|
||||||
+ <ref name="diskSourceNetworkProtocolVitastor"/>
|
|
||||||
</choice>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
diff --git a/include/libvirt/libvirt-storage.h b/include/libvirt/libvirt-storage.h
|
|
||||||
index 4bf2b5f..dbc011b 100644
|
|
||||||
--- a/include/libvirt/libvirt-storage.h
|
|
||||||
+++ b/include/libvirt/libvirt-storage.h
|
|
||||||
@@ -240,6 +240,7 @@ typedef enum {
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER = 1 << 16,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS = 1 << 17,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE = 1 << 18,
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR = 1 << 20,
|
|
||||||
} virConnectListAllStoragePoolsFlags;
|
|
||||||
|
|
||||||
int virConnectListAllStoragePools(virConnectPtr conn,
|
|
||||||
diff --git a/src/conf/domain_conf.c b/src/conf/domain_conf.c
|
|
||||||
index 222bb8c..2c30c55 100644
|
|
||||||
--- a/src/conf/domain_conf.c
|
|
||||||
+++ b/src/conf/domain_conf.c
|
|
||||||
@@ -4667,8 +4667,7 @@ virDomainDeviceDefPostParseCommon(virDomainDeviceDefPtr dev,
|
|
||||||
if (dev->type == VIR_DOMAIN_DEVICE_DISK) {
|
|
||||||
virDomainDiskDefPtr disk = dev->data.disk;
|
|
||||||
|
|
||||||
- /* internal snapshots and config files are currently supported
|
|
||||||
- * only with rbd: */
|
|
||||||
+ /* internal snapshots are currently supported only with rbd: */
|
|
||||||
if (virStorageSourceGetActualType(disk->src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
disk->src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD) {
|
|
||||||
if (disk->src->snapshot) {
|
|
||||||
@@ -4677,11 +4676,15 @@ virDomainDeviceDefPostParseCommon(virDomainDeviceDefPtr dev,
|
|
||||||
"only with 'rbd' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
-
|
|
||||||
+ }
|
|
||||||
+ /* config files are currently supported only with rbd and vitastor: */
|
|
||||||
+ if (virStorageSourceGetActualType(disk->src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
+ disk->src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD &&
|
|
||||||
+ disk->src->protocol != VIR_STORAGE_NET_PROTOCOL_VITASTOR) {
|
|
||||||
if (disk->src->configFile) {
|
|
||||||
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
_("<config> element is currently supported "
|
|
||||||
- "only with 'rbd' disks"));
|
|
||||||
+ "only with 'rbd' and 'vitastor' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -8653,6 +8656,10 @@ virDomainDiskSourceNetworkParse(xmlNodePtr node,
|
|
||||||
goto cleanup;
|
|
||||||
}
|
|
||||||
|
|
||||||
+ if (src->protocol == VIR_STORAGE_NET_PROTOCOL_VITASTOR) {
|
|
||||||
+ src->relPath = virXMLPropString(node, "query");
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
if ((haveTLS = virXMLPropString(node, "tls")) &&
|
|
||||||
(src->haveTLS = virTristateBoolTypeFromString(haveTLS)) <= 0) {
|
|
||||||
virReportError(VIR_ERR_XML_ERROR,
|
|
||||||
@@ -23849,6 +23856,10 @@ virDomainDiskSourceFormatNetwork(virBufferPtr attrBuf,
|
|
||||||
|
|
||||||
virBufferEscapeString(attrBuf, " name='%s'", path ? path : src->path);
|
|
||||||
|
|
||||||
+ if (src->protocol == VIR_STORAGE_NET_PROTOCOL_VITASTOR && src->relPath != NULL) {
|
|
||||||
+ virBufferEscapeString(attrBuf, " query='%s'", src->relPath);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
VIR_FREE(path);
|
|
||||||
|
|
||||||
if (src->haveTLS != VIR_TRISTATE_BOOL_ABSENT &&
|
|
||||||
@@ -30930,6 +30941,7 @@ virDomainDiskTranslateSourcePool(virDomainDiskDefPtr def)
|
|
||||||
|
|
||||||
case VIR_STORAGE_POOL_MPATH:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
diff --git a/src/conf/storage_conf.c b/src/conf/storage_conf.c
|
|
||||||
index 55db7a9..7cbe937 100644
|
|
||||||
--- a/src/conf/storage_conf.c
|
|
||||||
+++ b/src/conf/storage_conf.c
|
|
||||||
@@ -58,7 +58,7 @@ VIR_ENUM_IMPL(virStoragePool,
|
|
||||||
"logical", "disk", "iscsi",
|
|
||||||
"iscsi-direct", "scsi", "mpath",
|
|
||||||
"rbd", "sheepdog", "gluster",
|
|
||||||
- "zfs", "vstorage")
|
|
||||||
+ "zfs", "vstorage", "vitastor")
|
|
||||||
|
|
||||||
VIR_ENUM_IMPL(virStoragePoolFormatFileSystem,
|
|
||||||
VIR_STORAGE_POOL_FS_LAST,
|
|
||||||
@@ -232,6 +232,18 @@ static virStoragePoolTypeInfo poolTypeInfo[] = {
|
|
||||||
.formatToString = virStorageFileFormatTypeToString,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
+ {.poolType = VIR_STORAGE_POOL_VITASTOR,
|
|
||||||
+ .poolOptions = {
|
|
||||||
+ .flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NETWORK |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NAME),
|
|
||||||
+ },
|
|
||||||
+ .volOptions = {
|
|
||||||
+ .defaultFormat = VIR_STORAGE_FILE_RAW,
|
|
||||||
+ .formatFromString = virStorageVolumeFormatFromString,
|
|
||||||
+ .formatToString = virStorageFileFormatTypeToString,
|
|
||||||
+ }
|
|
||||||
+ },
|
|
||||||
{.poolType = VIR_STORAGE_POOL_SHEEPDOG,
|
|
||||||
.poolOptions = {
|
|
||||||
.flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
@@ -434,6 +446,11 @@ virStoragePoolDefParseSource(xmlXPathContextPtr ctxt,
|
|
||||||
_("element 'name' is mandatory for RBD pool"));
|
|
||||||
goto cleanup;
|
|
||||||
}
|
|
||||||
+ if (pool_type == VIR_STORAGE_POOL_VITASTOR && source->name == NULL) {
|
|
||||||
+ virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
+ _("element 'name' is mandatory for Vitastor pool"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
|
|
||||||
if (options->formatFromString) {
|
|
||||||
char *format = virXPathString("string(./format/@type)", ctxt);
|
|
||||||
@@ -1009,6 +1026,7 @@ virStoragePoolDefFormatBuf(virBufferPtr buf,
|
|
||||||
/* RBD, Sheepdog, Gluster and Iscsi-direct devices are not local block devs nor
|
|
||||||
* files, so they don't have a target */
|
|
||||||
if (def->type != VIR_STORAGE_POOL_RBD &&
|
|
||||||
+ def->type != VIR_STORAGE_POOL_VITASTOR &&
|
|
||||||
def->type != VIR_STORAGE_POOL_SHEEPDOG &&
|
|
||||||
def->type != VIR_STORAGE_POOL_GLUSTER &&
|
|
||||||
def->type != VIR_STORAGE_POOL_ISCSI_DIRECT) {
|
|
||||||
diff --git a/src/conf/storage_conf.h b/src/conf/storage_conf.h
|
|
||||||
index dc0aa2a..ed4983d 100644
|
|
||||||
--- a/src/conf/storage_conf.h
|
|
||||||
+++ b/src/conf/storage_conf.h
|
|
||||||
@@ -91,6 +91,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_POOL_GLUSTER, /* Gluster device */
|
|
||||||
VIR_STORAGE_POOL_ZFS, /* ZFS */
|
|
||||||
VIR_STORAGE_POOL_VSTORAGE, /* Virtuozzo Storage */
|
|
||||||
+ VIR_STORAGE_POOL_VITASTOR, /* Vitastor */
|
|
||||||
|
|
||||||
VIR_STORAGE_POOL_LAST,
|
|
||||||
} virStoragePoolType;
|
|
||||||
@@ -422,6 +423,7 @@ VIR_ENUM_DECL(virStoragePartedFs)
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SCSI | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_MPATH | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_RBD | \
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS | \
|
|
||||||
diff --git a/src/conf/virstorageobj.c b/src/conf/virstorageobj.c
|
|
||||||
index 6ea6a97..3ba45b9 100644
|
|
||||||
--- a/src/conf/virstorageobj.c
|
|
||||||
+++ b/src/conf/virstorageobj.c
|
|
||||||
@@ -1478,6 +1478,7 @@ virStoragePoolObjSourceFindDuplicateCb(const void *payload,
|
|
||||||
return 1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
@@ -1971,6 +1972,8 @@ virStoragePoolObjMatch(virStoragePoolObjPtr obj,
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_MPATH)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_RBD) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_RBD)) ||
|
|
||||||
+ (MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR) &&
|
|
||||||
+ (obj->def->type == VIR_STORAGE_POOL_VITASTOR)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_SHEEPDOG)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER) &&
|
|
||||||
diff --git a/src/libvirt-storage.c b/src/libvirt-storage.c
|
|
||||||
index 2ea3e94..d5d2273 100644
|
|
||||||
--- a/src/libvirt-storage.c
|
|
||||||
+++ b/src/libvirt-storage.c
|
|
||||||
@@ -92,6 +92,7 @@ virStoragePoolGetConnect(virStoragePoolPtr pool)
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SCSI
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_MPATH
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_RBD
|
|
||||||
+ * VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG
|
|
||||||
*
|
|
||||||
* Returns the number of storage pools found or -1 and sets @pools to
|
|
||||||
diff --git a/src/libxl/libxl_conf.c b/src/libxl/libxl_conf.c
|
|
||||||
index 73e988a..ab7bb81 100644
|
|
||||||
--- a/src/libxl/libxl_conf.c
|
|
||||||
+++ b/src/libxl/libxl_conf.c
|
|
||||||
@@ -905,6 +905,7 @@ libxlMakeNetworkDiskSrcStr(virStorageSourcePtr src,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
diff --git a/src/qemu/qemu_block.c b/src/qemu/qemu_block.c
|
|
||||||
index cbf0aa4..f0ca9e7 100644
|
|
||||||
--- a/src/qemu/qemu_block.c
|
|
||||||
+++ b/src/qemu/qemu_block.c
|
|
||||||
@@ -959,6 +959,42 @@ qemuBlockStorageSourceGetRBDProps(virStorageSourcePtr src)
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
+static virJSONValuePtr
|
|
||||||
+qemuBlockStorageSourceGetVitastorProps(virStorageSource *src)
|
|
||||||
+{
|
|
||||||
+ virJSONValuePtr ret = NULL;
|
|
||||||
+ virStorageNetHostDefPtr host;
|
|
||||||
+ size_t i;
|
|
||||||
+ virBuffer buf = VIR_BUFFER_INITIALIZER;
|
|
||||||
+ char *etcd = NULL;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ host = src->hosts + i;
|
|
||||||
+ if ((virStorageNetHostTransport)host->transport != VIR_STORAGE_NET_HOST_TRANS_TCP) {
|
|
||||||
+ goto cleanup;
|
|
||||||
+ }
|
|
||||||
+ virBufferAsprintf(&buf, i > 0 ? ",%s:%u" : "%s:%u", host->name, host->port);
|
|
||||||
+ }
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ etcd = virBufferContentAndReset(&buf);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virJSONValueObjectCreate(&ret,
|
|
||||||
+ "s:driver", "vitastor",
|
|
||||||
+ "S:etcd-host", etcd,
|
|
||||||
+ "S:etcd-prefix", src->relPath,
|
|
||||||
+ "S:config-path", src->configFile,
|
|
||||||
+ "s:image", src->path,
|
|
||||||
+ NULL) < 0)
|
|
||||||
+ goto cleanup;
|
|
||||||
+
|
|
||||||
+cleanup:
|
|
||||||
+ VIR_FREE(etcd);
|
|
||||||
+ virBufferFreeAndReset(&buf);
|
|
||||||
+ return ret;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
static virJSONValuePtr
|
|
||||||
qemuBlockStorageSourceGetSheepdogProps(virStorageSourcePtr src)
|
|
||||||
{
|
|
||||||
@@ -1174,6 +1210,11 @@ qemuBlockStorageSourceGetBackendProps(virStorageSourcePtr src,
|
|
||||||
return NULL;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (!(fileprops = qemuBlockStorageSourceGetVitastorProps(src)))
|
|
||||||
+ return NULL;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
if (!(fileprops = qemuBlockStorageSourceGetSheepdogProps(src)))
|
|
||||||
return NULL;
|
|
||||||
diff --git a/src/qemu/qemu_command.c b/src/qemu/qemu_command.c
|
|
||||||
index 822d5f8..abec34e 100644
|
|
||||||
--- a/src/qemu/qemu_command.c
|
|
||||||
+++ b/src/qemu/qemu_command.c
|
|
||||||
@@ -975,6 +975,43 @@ qemuBuildNetworkDriveStr(virStorageSourcePtr src,
|
|
||||||
ret = virBufferContentAndReset(&buf);
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (strchr(src->path, ':')) {
|
|
||||||
+ virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
||||||
+ _("':' not allowed in Vitastor source volume name '%s'"),
|
|
||||||
+ src->path);
|
|
||||||
+ return NULL;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ virBufferStrcat(&buf, "vitastor:image=", src->path, NULL);
|
|
||||||
+
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ virBufferAddLit(&buf, ":etcd-host=");
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ if (i)
|
|
||||||
+ virBufferAddLit(&buf, ",");
|
|
||||||
+
|
|
||||||
+ /* assume host containing : is ipv6 */
|
|
||||||
+ if (strchr(src->hosts[i].name, ':'))
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", "[%s]",
|
|
||||||
+ src->hosts[i].name);
|
|
||||||
+ else
|
|
||||||
+ virBufferAsprintf(&buf, "%s", src->hosts[i].name);
|
|
||||||
+
|
|
||||||
+ if (src->hosts[i].port)
|
|
||||||
+ virBufferAsprintf(&buf, "\\:%u", src->hosts[i].port);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (src->configFile)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":config-path=%s", src->configFile);
|
|
||||||
+
|
|
||||||
+ if (src->relPath)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":etcd-prefix=%s", src->relPath);
|
|
||||||
+
|
|
||||||
+ ret = virBufferContentAndReset(&buf);
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
||||||
_("VxHS protocol does not support URI syntax"));
|
|
||||||
diff --git a/src/qemu/qemu_domain.c b/src/qemu/qemu_domain.c
|
|
||||||
index ec6b340..f399efa 100644
|
|
||||||
--- a/src/qemu/qemu_domain.c
|
|
||||||
+++ b/src/qemu/qemu_domain.c
|
|
||||||
@@ -10881,6 +10881,7 @@ qemuDomainPrepareStorageSourceTLS(virStorageSourcePtr src,
|
|
||||||
break;
|
|
||||||
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
diff --git a/src/qemu/qemu_driver.c b/src/qemu/qemu_driver.c
|
|
||||||
index 1d96170..2d24396 100644
|
|
||||||
--- a/src/qemu/qemu_driver.c
|
|
||||||
+++ b/src/qemu/qemu_driver.c
|
|
||||||
@@ -14687,6 +14687,7 @@ qemuDomainSnapshotPrepareDiskExternalInactive(virDomainSnapshotDiskDefPtr snapdi
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_TFTP:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
||||||
_("external inactive snapshots are not supported on "
|
|
||||||
@@ -14764,6 +14765,7 @@ qemuDomainSnapshotPrepareDiskExternalActive(virDomainSnapshotDiskDefPtr snapdisk
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_TFTP:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
||||||
_("external active snapshots are not supported on "
|
|
||||||
@@ -14887,6 +14889,7 @@ qemuDomainSnapshotPrepareDiskInternal(virDomainDiskDefPtr disk,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_TFTP:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
||||||
_("internal inactive snapshots are not supported on "
|
|
||||||
diff --git a/src/qemu/qemu_parse_command.c b/src/qemu/qemu_parse_command.c
|
|
||||||
index c4650f0..551da41 100644
|
|
||||||
--- a/src/qemu/qemu_parse_command.c
|
|
||||||
+++ b/src/qemu/qemu_parse_command.c
|
|
||||||
@@ -2184,6 +2184,7 @@ qemuParseCommandLine(virFileCachePtr capsCache,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_TFTP:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
/* ignored for now */
|
|
||||||
break;
|
|
||||||
diff --git a/src/storage/storage_driver.c b/src/storage/storage_driver.c
|
|
||||||
index 4a13e90..33301c7 100644
|
|
||||||
--- a/src/storage/storage_driver.c
|
|
||||||
+++ b/src/storage/storage_driver.c
|
|
||||||
@@ -1568,6 +1568,7 @@ storageVolLookupByPathCallback(virStoragePoolObjPtr obj,
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_ZFS:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
ignore_value(VIR_STRDUP(stable_path, data->path));
|
|
||||||
break;
|
|
||||||
diff --git a/src/util/virstoragefile.c b/src/util/virstoragefile.c
|
|
||||||
index bd4b027..8454906 100644
|
|
||||||
--- a/src/util/virstoragefile.c
|
|
||||||
+++ b/src/util/virstoragefile.c
|
|
||||||
@@ -84,7 +84,8 @@ VIR_ENUM_IMPL(virStorageNetProtocol, VIR_STORAGE_NET_PROTOCOL_LAST,
|
|
||||||
"ftps",
|
|
||||||
"tftp",
|
|
||||||
"ssh",
|
|
||||||
- "vxhs")
|
|
||||||
+ "vxhs",
|
|
||||||
+ "vitastor")
|
|
||||||
|
|
||||||
VIR_ENUM_IMPL(virStorageNetHostTransport, VIR_STORAGE_NET_HOST_TRANS_LAST,
|
|
||||||
"tcp",
|
|
||||||
@@ -2839,6 +2840,83 @@ virStorageSourceParseRBDColonString(const char *rbdstr,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
+static int
|
|
||||||
+virStorageSourceParseVitastorColonString(const char *colonstr,
|
|
||||||
+ virStorageSourcePtr src)
|
|
||||||
+{
|
|
||||||
+ char *p, *e, *next;
|
|
||||||
+ char *options = NULL;
|
|
||||||
+
|
|
||||||
+ /* optionally skip the "vitastor:" prefix if provided */
|
|
||||||
+ if (STRPREFIX(colonstr, "vitastor:"))
|
|
||||||
+ colonstr += strlen("vitastor:");
|
|
||||||
+
|
|
||||||
+ if (VIR_STRDUP(options, colonstr) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+
|
|
||||||
+ p = options;
|
|
||||||
+ while (*p) {
|
|
||||||
+ /* find : delimiter or end of string */
|
|
||||||
+ for (e = p; *e && *e != ':'; ++e) {
|
|
||||||
+ if (*e == '\\') {
|
|
||||||
+ e++;
|
|
||||||
+ if (*e == '\0')
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ if (*e == '\0') {
|
|
||||||
+ next = e; /* last kv pair */
|
|
||||||
+ } else {
|
|
||||||
+ next = e + 1;
|
|
||||||
+ *e = '\0';
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (STRPREFIX(p, "image=")) {
|
|
||||||
+ if (VIR_STRDUP(src->path, p + strlen("image=")) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-prefix=")) {
|
|
||||||
+ if (VIR_STRDUP(src->relPath, p + strlen("etcd-prefix=")) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ } else if (STRPREFIX(p, "config-path=")) {
|
|
||||||
+ if (VIR_STRDUP(src->configFile, p + strlen("config-path=")) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-host=")) {
|
|
||||||
+ char *h, *sep;
|
|
||||||
+
|
|
||||||
+ h = p + strlen("etcd-host=");
|
|
||||||
+ while (h < e) {
|
|
||||||
+ for (sep = h; sep < e; ++sep) {
|
|
||||||
+ if (*sep == '\\' && (sep[1] == ',' ||
|
|
||||||
+ sep[1] == ';' ||
|
|
||||||
+ sep[1] == ' ')) {
|
|
||||||
+ *sep = '\0';
|
|
||||||
+ sep += 2;
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virStorageSourceRBDAddHost(src, h) < 0)
|
|
||||||
+ goto error;
|
|
||||||
+
|
|
||||||
+ h = sep;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ p = next;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (!src->path) {
|
|
||||||
+ goto error;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+
|
|
||||||
+error:
|
|
||||||
+ VIR_FREE(options);
|
|
||||||
+ return -1;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
static int
|
|
||||||
virStorageSourceParseNBDColonString(const char *nbdstr,
|
|
||||||
virStorageSourcePtr src)
|
|
||||||
@@ -2942,6 +3020,11 @@ virStorageSourceParseBackingColon(virStorageSourcePtr src,
|
|
||||||
goto cleanup;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (virStorageSourceParseVitastorColonString(path, src) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
@@ -3441,6 +3524,56 @@ virStorageSourceParseBackingJSONRBD(virStorageSourcePtr src,
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
+static int
|
|
||||||
+virStorageSourceParseBackingJSONVitastor(virStorageSourcePtr src,
|
|
||||||
+ virJSONValuePtr json,
|
|
||||||
+ int opaque ATTRIBUTE_UNUSED)
|
|
||||||
+{
|
|
||||||
+ const char *filename;
|
|
||||||
+ const char *image = virJSONValueObjectGetString(json, "image");
|
|
||||||
+ const char *conf = virJSONValueObjectGetString(json, "config-path");
|
|
||||||
+ const char *etcd_prefix = virJSONValueObjectGetString(json, "etcd-prefix");
|
|
||||||
+ virJSONValuePtr servers = virJSONValueObjectGetArray(json, "server");
|
|
||||||
+ size_t nservers;
|
|
||||||
+ size_t i;
|
|
||||||
+
|
|
||||||
+ src->type = VIR_STORAGE_TYPE_NETWORK;
|
|
||||||
+ src->protocol = VIR_STORAGE_NET_PROTOCOL_VITASTOR;
|
|
||||||
+
|
|
||||||
+ /* legacy syntax passed via 'filename' option */
|
|
||||||
+ if ((filename = virJSONValueObjectGetString(json, "filename")))
|
|
||||||
+ return virStorageSourceParseVitastorColonString(filename, src);
|
|
||||||
+
|
|
||||||
+ if (!image) {
|
|
||||||
+ virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
||||||
+ _("missing image name in Vitastor backing volume "
|
|
||||||
+ "JSON specification"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (VIR_STRDUP(src->path, image) < 0 ||
|
|
||||||
+ VIR_STRDUP(src->configFile, conf) < 0 ||
|
|
||||||
+ VIR_STRDUP(src->relPath, etcd_prefix) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+
|
|
||||||
+ if (servers) {
|
|
||||||
+ nservers = virJSONValueArraySize(servers);
|
|
||||||
+
|
|
||||||
+ if (VIR_ALLOC_N(src->hosts, nservers) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+
|
|
||||||
+ src->nhosts = nservers;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < nservers; i++) {
|
|
||||||
+ if (virStorageSourceParseBackingJSONInetSocketAddress(src->hosts + i,
|
|
||||||
+ virJSONValueArrayGet(servers, i)) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
static int
|
|
||||||
virStorageSourceParseBackingJSONRaw(virStorageSourcePtr src,
|
|
||||||
virJSONValuePtr json,
|
|
||||||
@@ -3507,6 +3640,7 @@ static const struct virStorageSourceJSONDriverParser jsonParsers[] = {
|
|
||||||
{"sheepdog", virStorageSourceParseBackingJSONSheepdog, 0},
|
|
||||||
{"ssh", virStorageSourceParseBackingJSONSSH, 0},
|
|
||||||
{"rbd", virStorageSourceParseBackingJSONRBD, 0},
|
|
||||||
+ {"vitastor", virStorageSourceParseBackingJSONVitastor, 0},
|
|
||||||
{"raw", virStorageSourceParseBackingJSONRaw, 0},
|
|
||||||
{"vxhs", virStorageSourceParseBackingJSONVxHS, 0},
|
|
||||||
};
|
|
||||||
@@ -4276,6 +4410,7 @@ virStorageSourceNetworkDefaultPort(virStorageNetProtocol protocol)
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
return 24007;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
/* we don't provide a default for RBD */
|
|
||||||
return 0;
|
|
||||||
diff --git a/src/util/virstoragefile.h b/src/util/virstoragefile.h
|
|
||||||
index 1d6161a..8d83bf3 100644
|
|
||||||
--- a/src/util/virstoragefile.h
|
|
||||||
+++ b/src/util/virstoragefile.h
|
|
||||||
@@ -134,6 +134,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_TFTP,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_SSH,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_VXHS,
|
|
||||||
+ VIR_STORAGE_NET_PROTOCOL_VITASTOR,
|
|
||||||
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_LAST
|
|
||||||
} virStorageNetProtocol;
|
|
||||||
diff --git a/src/xenconfig/xen_xl.c b/src/xenconfig/xen_xl.c
|
|
||||||
index accfc3a..a18f9c3 100644
|
|
||||||
--- a/src/xenconfig/xen_xl.c
|
|
||||||
+++ b/src/xenconfig/xen_xl.c
|
|
||||||
@@ -1535,6 +1535,7 @@ xenFormatXLDiskSrcNet(virStorageSourcePtr src)
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
diff --git a/tools/virsh-pool.c b/tools/virsh-pool.c
|
|
||||||
index 70ca39b..9caef51 100644
|
|
||||||
--- a/tools/virsh-pool.c
|
|
||||||
+++ b/tools/virsh-pool.c
|
|
||||||
@@ -1219,6 +1219,9 @@ cmdPoolList(vshControl *ctl, const vshCmd *cmd ATTRIBUTE_UNUSED)
|
|
||||||
case VIR_STORAGE_POOL_VSTORAGE:
|
|
||||||
flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE;
|
|
||||||
break;
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
+ flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR;
|
|
||||||
+ break;
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
}
|
|
|
@ -1,685 +0,0 @@
|
||||||
commit 4e74c622884e2585b2cfcdf322fbd2bff6de41ca
|
|
||||||
Author: Vitaliy Filippov <vitalif@yourcmc.ru>
|
|
||||||
Date: Fri Jul 9 01:31:57 2021 +0300
|
|
||||||
|
|
||||||
Add Vitastor support
|
|
||||||
|
|
||||||
diff --git a/docs/schemas/domaincommon.rng b/docs/schemas/domaincommon.rng
|
|
||||||
index 7dc419b..875433b 100644
|
|
||||||
--- a/docs/schemas/domaincommon.rng
|
|
||||||
+++ b/docs/schemas/domaincommon.rng
|
|
||||||
@@ -1827,6 +1827,35 @@
|
|
||||||
</element>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
+ <define name="diskSourceNetworkProtocolVitastor">
|
|
||||||
+ <element name="source">
|
|
||||||
+ <interleave>
|
|
||||||
+ <attribute name="protocol">
|
|
||||||
+ <value>vitastor</value>
|
|
||||||
+ </attribute>
|
|
||||||
+ <ref name="diskSourceCommon"/>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="name"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="query"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <zeroOrMore>
|
|
||||||
+ <ref name="diskSourceNetworkHost"/>
|
|
||||||
+ </zeroOrMore>
|
|
||||||
+ <optional>
|
|
||||||
+ <element name="config">
|
|
||||||
+ <attribute name="file">
|
|
||||||
+ <ref name="absFilePath"/>
|
|
||||||
+ </attribute>
|
|
||||||
+ <empty/>
|
|
||||||
+ </element>
|
|
||||||
+ </optional>
|
|
||||||
+ <empty/>
|
|
||||||
+ </interleave>
|
|
||||||
+ </element>
|
|
||||||
+ </define>
|
|
||||||
+
|
|
||||||
<define name="diskSourceNetworkProtocolISCSI">
|
|
||||||
<element name="source">
|
|
||||||
<attribute name="protocol">
|
|
||||||
@@ -2083,6 +2112,7 @@
|
|
||||||
<ref name="diskSourceNetworkProtocolSimple"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolVxHS"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolNFS"/>
|
|
||||||
+ <ref name="diskSourceNetworkProtocolVitastor"/>
|
|
||||||
</choice>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
diff --git a/include/libvirt/libvirt-storage.h b/include/libvirt/libvirt-storage.h
|
|
||||||
index 089e1e0..d7e7ef4 100644
|
|
||||||
--- a/include/libvirt/libvirt-storage.h
|
|
||||||
+++ b/include/libvirt/libvirt-storage.h
|
|
||||||
@@ -245,6 +245,7 @@ typedef enum {
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS = 1 << 17,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE = 1 << 18,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ISCSI_DIRECT = 1 << 19,
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR = 1 << 20,
|
|
||||||
} virConnectListAllStoragePoolsFlags;
|
|
||||||
|
|
||||||
int virConnectListAllStoragePools(virConnectPtr conn,
|
|
||||||
diff --git a/src/conf/domain_conf.c b/src/conf/domain_conf.c
|
|
||||||
index 01b7187..645c758 100644
|
|
||||||
--- a/src/conf/domain_conf.c
|
|
||||||
+++ b/src/conf/domain_conf.c
|
|
||||||
@@ -5230,8 +5230,7 @@ virDomainDiskDefPostParse(virDomainDiskDefPtr disk,
|
|
||||||
const virDomainDef *def,
|
|
||||||
virDomainXMLOptionPtr xmlopt)
|
|
||||||
{
|
|
||||||
- /* internal snapshots and config files are currently supported
|
|
||||||
- * only with rbd: */
|
|
||||||
+ /* internal snapshots are currently supported only with rbd: */
|
|
||||||
if (virStorageSourceGetActualType(disk->src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
disk->src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD) {
|
|
||||||
if (disk->src->snapshot) {
|
|
||||||
@@ -5240,11 +5239,15 @@ virDomainDiskDefPostParse(virDomainDiskDefPtr disk,
|
|
||||||
"only with 'rbd' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
-
|
|
||||||
+ }
|
|
||||||
+ /* config files are currently supported only with rbd and vitastor: */
|
|
||||||
+ if (virStorageSourceGetActualType(disk->src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
+ disk->src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD &&
|
|
||||||
+ disk->src->protocol != VIR_STORAGE_NET_PROTOCOL_VITASTOR) {
|
|
||||||
if (disk->src->configFile) {
|
|
||||||
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
_("<config> element is currently supported "
|
|
||||||
- "only with 'rbd' disks"));
|
|
||||||
+ "only with 'rbd' and 'vitastor' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -8261,7 +8264,8 @@ virDomainDiskSourceNetworkParse(xmlNodePtr node,
|
|
||||||
src->configFile = virXPathString("string(./config/@file)", ctxt);
|
|
||||||
|
|
||||||
if (src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTP ||
|
|
||||||
- src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTPS)
|
|
||||||
+ src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTPS ||
|
|
||||||
+ src->protocol == VIR_STORAGE_NET_PROTOCOL_VITASTOR)
|
|
||||||
src->query = virXMLPropString(node, "query");
|
|
||||||
|
|
||||||
if (virDomainStorageNetworkParseHosts(node, ctxt, &src->hosts, &src->nhosts) < 0)
|
|
||||||
@@ -31392,6 +31396,7 @@ virDomainStorageSourceTranslateSourcePool(virStorageSourcePtr src,
|
|
||||||
|
|
||||||
case VIR_STORAGE_POOL_MPATH:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
diff --git a/src/conf/storage_conf.c b/src/conf/storage_conf.c
|
|
||||||
index 0c50529..fe97574 100644
|
|
||||||
--- a/src/conf/storage_conf.c
|
|
||||||
+++ b/src/conf/storage_conf.c
|
|
||||||
@@ -60,7 +60,7 @@ VIR_ENUM_IMPL(virStoragePool,
|
|
||||||
"logical", "disk", "iscsi",
|
|
||||||
"iscsi-direct", "scsi", "mpath",
|
|
||||||
"rbd", "sheepdog", "gluster",
|
|
||||||
- "zfs", "vstorage",
|
|
||||||
+ "zfs", "vstorage", "vitastor",
|
|
||||||
);
|
|
||||||
|
|
||||||
VIR_ENUM_IMPL(virStoragePoolFormatFileSystem,
|
|
||||||
@@ -249,6 +249,18 @@ static virStoragePoolTypeInfo poolTypeInfo[] = {
|
|
||||||
.formatToString = virStorageFileFormatTypeToString,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
+ {.poolType = VIR_STORAGE_POOL_VITASTOR,
|
|
||||||
+ .poolOptions = {
|
|
||||||
+ .flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NETWORK |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NAME),
|
|
||||||
+ },
|
|
||||||
+ .volOptions = {
|
|
||||||
+ .defaultFormat = VIR_STORAGE_FILE_RAW,
|
|
||||||
+ .formatFromString = virStorageVolumeFormatFromString,
|
|
||||||
+ .formatToString = virStorageFileFormatTypeToString,
|
|
||||||
+ }
|
|
||||||
+ },
|
|
||||||
{.poolType = VIR_STORAGE_POOL_SHEEPDOG,
|
|
||||||
.poolOptions = {
|
|
||||||
.flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
@@ -551,6 +563,11 @@ virStoragePoolDefParseSource(xmlXPathContextPtr ctxt,
|
|
||||||
_("element 'name' is mandatory for RBD pool"));
|
|
||||||
goto cleanup;
|
|
||||||
}
|
|
||||||
+ if (pool_type == VIR_STORAGE_POOL_VITASTOR && source->name == NULL) {
|
|
||||||
+ virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
+ _("element 'name' is mandatory for Vitastor pool"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
|
|
||||||
if (options->formatFromString) {
|
|
||||||
g_autofree char *format = NULL;
|
|
||||||
@@ -1217,6 +1234,7 @@ virStoragePoolDefFormatBuf(virBufferPtr buf,
|
|
||||||
/* RBD, Sheepdog, Gluster and Iscsi-direct devices are not local block devs nor
|
|
||||||
* files, so they don't have a target */
|
|
||||||
if (def->type != VIR_STORAGE_POOL_RBD &&
|
|
||||||
+ def->type != VIR_STORAGE_POOL_VITASTOR &&
|
|
||||||
def->type != VIR_STORAGE_POOL_SHEEPDOG &&
|
|
||||||
def->type != VIR_STORAGE_POOL_GLUSTER &&
|
|
||||||
def->type != VIR_STORAGE_POOL_ISCSI_DIRECT) {
|
|
||||||
diff --git a/src/conf/storage_conf.h b/src/conf/storage_conf.h
|
|
||||||
index ffd406e..8868a05 100644
|
|
||||||
--- a/src/conf/storage_conf.h
|
|
||||||
+++ b/src/conf/storage_conf.h
|
|
||||||
@@ -110,6 +110,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_POOL_GLUSTER, /* Gluster device */
|
|
||||||
VIR_STORAGE_POOL_ZFS, /* ZFS */
|
|
||||||
VIR_STORAGE_POOL_VSTORAGE, /* Virtuozzo Storage */
|
|
||||||
+ VIR_STORAGE_POOL_VITASTOR, /* Vitastor */
|
|
||||||
|
|
||||||
VIR_STORAGE_POOL_LAST,
|
|
||||||
} virStoragePoolType;
|
|
||||||
@@ -474,6 +475,7 @@ VIR_ENUM_DECL(virStoragePartedFs);
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SCSI | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_MPATH | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_RBD | \
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS | \
|
|
||||||
diff --git a/src/conf/virstorageobj.c b/src/conf/virstorageobj.c
|
|
||||||
index 9fe8b3f..bf595b0 100644
|
|
||||||
--- a/src/conf/virstorageobj.c
|
|
||||||
+++ b/src/conf/virstorageobj.c
|
|
||||||
@@ -1491,6 +1491,7 @@ virStoragePoolObjSourceFindDuplicateCb(const void *payload,
|
|
||||||
return 1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
@@ -1990,6 +1991,8 @@ virStoragePoolObjMatch(virStoragePoolObjPtr obj,
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_MPATH)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_RBD) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_RBD)) ||
|
|
||||||
+ (MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR) &&
|
|
||||||
+ (obj->def->type == VIR_STORAGE_POOL_VITASTOR)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_SHEEPDOG)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER) &&
|
|
||||||
diff --git a/src/libvirt-storage.c b/src/libvirt-storage.c
|
|
||||||
index 2a7cdca..f756be1 100644
|
|
||||||
--- a/src/libvirt-storage.c
|
|
||||||
+++ b/src/libvirt-storage.c
|
|
||||||
@@ -92,6 +92,7 @@ virStoragePoolGetConnect(virStoragePoolPtr pool)
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SCSI
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_MPATH
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_RBD
|
|
||||||
+ * VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_ZFS
|
|
||||||
diff --git a/src/libxl/libxl_conf.c b/src/libxl/libxl_conf.c
|
|
||||||
index 6a8ae27..a735bc6 100644
|
|
||||||
--- a/src/libxl/libxl_conf.c
|
|
||||||
+++ b/src/libxl/libxl_conf.c
|
|
||||||
@@ -942,6 +942,7 @@ libxlMakeNetworkDiskSrcStr(virStorageSourcePtr src,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
diff --git a/src/libxl/xen_xl.c b/src/libxl/xen_xl.c
|
|
||||||
index 17b93d0..c5a0084 100644
|
|
||||||
--- a/src/libxl/xen_xl.c
|
|
||||||
+++ b/src/libxl/xen_xl.c
|
|
||||||
@@ -1601,6 +1601,7 @@ xenFormatXLDiskSrcNet(virStorageSourcePtr src)
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
diff --git a/src/qemu/qemu_block.c b/src/qemu/qemu_block.c
|
|
||||||
index f9c6da2..d837a05 100644
|
|
||||||
--- a/src/qemu/qemu_block.c
|
|
||||||
+++ b/src/qemu/qemu_block.c
|
|
||||||
@@ -938,6 +938,38 @@ qemuBlockStorageSourceGetRBDProps(virStorageSourcePtr src,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
+static virJSONValuePtr
|
|
||||||
+qemuBlockStorageSourceGetVitastorProps(virStorageSource *src)
|
|
||||||
+{
|
|
||||||
+ virJSONValuePtr ret = NULL;
|
|
||||||
+ virStorageNetHostDefPtr host;
|
|
||||||
+ size_t i;
|
|
||||||
+ g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
||||||
+ g_autofree char *etcd = NULL;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ host = src->hosts + i;
|
|
||||||
+ if ((virStorageNetHostTransport)host->transport != VIR_STORAGE_NET_HOST_TRANS_TCP) {
|
|
||||||
+ return NULL;
|
|
||||||
+ }
|
|
||||||
+ virBufferAsprintf(&buf, i > 0 ? ",%s:%u" : "%s:%u", host->name, host->port);
|
|
||||||
+ }
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ etcd = virBufferContentAndReset(&buf);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virJSONValueObjectCreate(&ret,
|
|
||||||
+ "S:etcd-host", etcd,
|
|
||||||
+ "S:etcd-prefix", src->query,
|
|
||||||
+ "S:config-path", src->configFile,
|
|
||||||
+ "s:image", src->path,
|
|
||||||
+ NULL) < 0)
|
|
||||||
+ return NULL;
|
|
||||||
+
|
|
||||||
+ return ret;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
static virJSONValuePtr
|
|
||||||
qemuBlockStorageSourceGetSheepdogProps(virStorageSourcePtr src)
|
|
||||||
{
|
|
||||||
@@ -1224,6 +1256,12 @@ qemuBlockStorageSourceGetBackendProps(virStorageSourcePtr src,
|
|
||||||
return NULL;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ driver = "vitastor";
|
|
||||||
+ if (!(fileprops = qemuBlockStorageSourceGetVitastorProps(src)))
|
|
||||||
+ return NULL;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
driver = "sheepdog";
|
|
||||||
if (!(fileprops = qemuBlockStorageSourceGetSheepdogProps(src)))
|
|
||||||
@@ -2183,6 +2221,7 @@ qemuBlockGetBackingStoreString(virStorageSourcePtr src,
|
|
||||||
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
@@ -2560,6 +2599,12 @@ qemuBlockStorageSourceCreateGetStorageProps(virStorageSourcePtr src,
|
|
||||||
return -1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ driver = "vitastor";
|
|
||||||
+ if (!(location = qemuBlockStorageSourceGetVitastorProps(src)))
|
|
||||||
+ return -1;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
driver = "sheepdog";
|
|
||||||
if (!(location = qemuBlockStorageSourceGetSheepdogProps(src)))
|
|
||||||
diff --git a/src/qemu/qemu_command.c b/src/qemu/qemu_command.c
|
|
||||||
index 6f970a3..4c03fb8 100644
|
|
||||||
--- a/src/qemu/qemu_command.c
|
|
||||||
+++ b/src/qemu/qemu_command.c
|
|
||||||
@@ -1034,6 +1034,43 @@ qemuBuildNetworkDriveStr(virStorageSourcePtr src,
|
|
||||||
ret = virBufferContentAndReset(&buf);
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (strchr(src->path, ':')) {
|
|
||||||
+ virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
||||||
+ _("':' not allowed in Vitastor source volume name '%s'"),
|
|
||||||
+ src->path);
|
|
||||||
+ return NULL;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ virBufferStrcat(&buf, "vitastor:image=", src->path, NULL);
|
|
||||||
+
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ virBufferAddLit(&buf, ":etcd-host=");
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ if (i)
|
|
||||||
+ virBufferAddLit(&buf, ",");
|
|
||||||
+
|
|
||||||
+ /* assume host containing : is ipv6 */
|
|
||||||
+ if (strchr(src->hosts[i].name, ':'))
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", "[%s]",
|
|
||||||
+ src->hosts[i].name);
|
|
||||||
+ else
|
|
||||||
+ virBufferAsprintf(&buf, "%s", src->hosts[i].name);
|
|
||||||
+
|
|
||||||
+ if (src->hosts[i].port)
|
|
||||||
+ virBufferAsprintf(&buf, "\\:%u", src->hosts[i].port);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (src->configFile)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":config-path=%s", src->configFile);
|
|
||||||
+
|
|
||||||
+ if (src->query)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":etcd-prefix=%s", src->query);
|
|
||||||
+
|
|
||||||
+ ret = virBufferContentAndReset(&buf);
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
||||||
_("VxHS protocol does not support URI syntax"));
|
|
||||||
diff --git a/src/qemu/qemu_domain.c b/src/qemu/qemu_domain.c
|
|
||||||
index 0765dc7..4cff344 100644
|
|
||||||
--- a/src/qemu/qemu_domain.c
|
|
||||||
+++ b/src/qemu/qemu_domain.c
|
|
||||||
@@ -4610,7 +4610,8 @@ qemuDomainValidateStorageSource(virStorageSourcePtr src,
|
|
||||||
if (src->query &&
|
|
||||||
(actualType != VIR_STORAGE_TYPE_NETWORK ||
|
|
||||||
(src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTPS &&
|
|
||||||
- src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTP))) {
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTP &&
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_VITASTOR))) {
|
|
||||||
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
||||||
_("query is supported only with HTTP(S) protocols"));
|
|
||||||
return -1;
|
|
||||||
@@ -9704,6 +9705,7 @@ qemuDomainPrepareStorageSourceTLS(virStorageSourcePtr src,
|
|
||||||
break;
|
|
||||||
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
diff --git a/src/qemu/qemu_snapshot.c b/src/qemu/qemu_snapshot.c
|
|
||||||
index ee333c3..674aa58 100644
|
|
||||||
--- a/src/qemu/qemu_snapshot.c
|
|
||||||
+++ b/src/qemu/qemu_snapshot.c
|
|
||||||
@@ -403,6 +403,7 @@ qemuSnapshotPrepareDiskExternalInactive(virDomainSnapshotDiskDefPtr snapdisk,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
@@ -493,6 +494,7 @@ qemuSnapshotPrepareDiskExternalActive(virDomainObjPtr vm,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_HTTP:
|
|
||||||
@@ -623,6 +625,7 @@ qemuSnapshotPrepareDiskInternal(virDomainDiskDefPtr disk,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
diff --git a/src/storage/storage_driver.c b/src/storage/storage_driver.c
|
|
||||||
index 16bc53a..1e5d820 100644
|
|
||||||
--- a/src/storage/storage_driver.c
|
|
||||||
+++ b/src/storage/storage_driver.c
|
|
||||||
@@ -1645,6 +1645,7 @@ storageVolLookupByPathCallback(virStoragePoolObjPtr obj,
|
|
||||||
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_ZFS:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
diff --git a/src/test/test_driver.c b/src/test/test_driver.c
|
|
||||||
index 29c4c86..a27ad94 100644
|
|
||||||
--- a/src/test/test_driver.c
|
|
||||||
+++ b/src/test/test_driver.c
|
|
||||||
@@ -7096,6 +7096,7 @@ testStorageVolumeTypeForPool(int pooltype)
|
|
||||||
case VIR_STORAGE_POOL_ISCSI_DIRECT:
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
return VIR_STORAGE_VOL_NETWORK;
|
|
||||||
case VIR_STORAGE_POOL_LOGICAL:
|
|
||||||
case VIR_STORAGE_POOL_DISK:
|
|
||||||
diff --git a/src/util/virstoragefile.c b/src/util/virstoragefile.c
|
|
||||||
index 0d3c2af..edb7f9e 100644
|
|
||||||
--- a/src/util/virstoragefile.c
|
|
||||||
+++ b/src/util/virstoragefile.c
|
|
||||||
@@ -91,6 +91,7 @@ VIR_ENUM_IMPL(virStorageNetProtocol,
|
|
||||||
"ssh",
|
|
||||||
"vxhs",
|
|
||||||
"nfs",
|
|
||||||
+ "vitastor",
|
|
||||||
);
|
|
||||||
|
|
||||||
VIR_ENUM_IMPL(virStorageNetHostTransport,
|
|
||||||
@@ -2880,6 +2881,75 @@ virStorageSourceParseRBDColonString(const char *rbdstr,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
+static int
|
|
||||||
+virStorageSourceParseVitastorColonString(const char *colonstr,
|
|
||||||
+ virStorageSourcePtr src)
|
|
||||||
+{
|
|
||||||
+ char *p, *e, *next;
|
|
||||||
+ g_autofree char *options = NULL;
|
|
||||||
+
|
|
||||||
+ /* optionally skip the "vitastor:" prefix if provided */
|
|
||||||
+ if (STRPREFIX(colonstr, "vitastor:"))
|
|
||||||
+ colonstr += strlen("vitastor:");
|
|
||||||
+
|
|
||||||
+ options = g_strdup(colonstr);
|
|
||||||
+
|
|
||||||
+ p = options;
|
|
||||||
+ while (*p) {
|
|
||||||
+ /* find : delimiter or end of string */
|
|
||||||
+ for (e = p; *e && *e != ':'; ++e) {
|
|
||||||
+ if (*e == '\\') {
|
|
||||||
+ e++;
|
|
||||||
+ if (*e == '\0')
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ if (*e == '\0') {
|
|
||||||
+ next = e; /* last kv pair */
|
|
||||||
+ } else {
|
|
||||||
+ next = e + 1;
|
|
||||||
+ *e = '\0';
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (STRPREFIX(p, "image=")) {
|
|
||||||
+ src->path = g_strdup(p + strlen("image="));
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-prefix=")) {
|
|
||||||
+ src->query = g_strdup(p + strlen("etcd-prefix="));
|
|
||||||
+ } else if (STRPREFIX(p, "config-path=")) {
|
|
||||||
+ src->configFile = g_strdup(p + strlen("config-path="));
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-host=")) {
|
|
||||||
+ char *h, *sep;
|
|
||||||
+
|
|
||||||
+ h = p + strlen("etcd-host=");
|
|
||||||
+ while (h < e) {
|
|
||||||
+ for (sep = h; sep < e; ++sep) {
|
|
||||||
+ if (*sep == '\\' && (sep[1] == ',' ||
|
|
||||||
+ sep[1] == ';' ||
|
|
||||||
+ sep[1] == ' ')) {
|
|
||||||
+ *sep = '\0';
|
|
||||||
+ sep += 2;
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virStorageSourceRBDAddHost(src, h) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+
|
|
||||||
+ h = sep;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ p = next;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (!src->path) {
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
static int
|
|
||||||
virStorageSourceParseNBDColonString(const char *nbdstr,
|
|
||||||
virStorageSourcePtr src)
|
|
||||||
@@ -2992,6 +3062,11 @@ virStorageSourceParseBackingColon(virStorageSourcePtr src,
|
|
||||||
return -1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (virStorageSourceParseVitastorColonString(path, src) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
@@ -3581,6 +3656,54 @@ virStorageSourceParseBackingJSONRBD(virStorageSourcePtr src,
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
+static int
|
|
||||||
+virStorageSourceParseBackingJSONVitastor(virStorageSourcePtr src,
|
|
||||||
+ virJSONValuePtr json,
|
|
||||||
+ const char *jsonstr G_GNUC_UNUSED,
|
|
||||||
+ int opaque G_GNUC_UNUSED)
|
|
||||||
+{
|
|
||||||
+ const char *filename;
|
|
||||||
+ const char *image = virJSONValueObjectGetString(json, "image");
|
|
||||||
+ const char *conf = virJSONValueObjectGetString(json, "config-path");
|
|
||||||
+ const char *etcd_prefix = virJSONValueObjectGetString(json, "etcd-prefix");
|
|
||||||
+ virJSONValuePtr servers = virJSONValueObjectGetArray(json, "server");
|
|
||||||
+ size_t nservers;
|
|
||||||
+ size_t i;
|
|
||||||
+
|
|
||||||
+ src->type = VIR_STORAGE_TYPE_NETWORK;
|
|
||||||
+ src->protocol = VIR_STORAGE_NET_PROTOCOL_VITASTOR;
|
|
||||||
+
|
|
||||||
+ /* legacy syntax passed via 'filename' option */
|
|
||||||
+ if ((filename = virJSONValueObjectGetString(json, "filename")))
|
|
||||||
+ return virStorageSourceParseVitastorColonString(filename, src);
|
|
||||||
+
|
|
||||||
+ if (!image) {
|
|
||||||
+ virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
||||||
+ _("missing image name in Vitastor backing volume "
|
|
||||||
+ "JSON specification"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ src->path = g_strdup(image);
|
|
||||||
+ src->configFile = g_strdup(conf);
|
|
||||||
+ src->query = g_strdup(etcd_prefix);
|
|
||||||
+
|
|
||||||
+ if (servers) {
|
|
||||||
+ nservers = virJSONValueArraySize(servers);
|
|
||||||
+
|
|
||||||
+ src->hosts = g_new0(virStorageNetHostDef, nservers);
|
|
||||||
+ src->nhosts = nservers;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < nservers; i++) {
|
|
||||||
+ if (virStorageSourceParseBackingJSONInetSocketAddress(src->hosts + i,
|
|
||||||
+ virJSONValueArrayGet(servers, i)) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
static int
|
|
||||||
virStorageSourceParseBackingJSONRaw(virStorageSourcePtr src,
|
|
||||||
virJSONValuePtr json,
|
|
||||||
@@ -3759,6 +3882,7 @@ static const struct virStorageSourceJSONDriverParser jsonParsers[] = {
|
|
||||||
{"sheepdog", false, virStorageSourceParseBackingJSONSheepdog, 0},
|
|
||||||
{"ssh", false, virStorageSourceParseBackingJSONSSH, 0},
|
|
||||||
{"rbd", false, virStorageSourceParseBackingJSONRBD, 0},
|
|
||||||
+ {"vitastor", false, virStorageSourceParseBackingJSONVitastor, 0},
|
|
||||||
{"raw", true, virStorageSourceParseBackingJSONRaw, 0},
|
|
||||||
{"nfs", false, virStorageSourceParseBackingJSONNFS, 0},
|
|
||||||
{"vxhs", false, virStorageSourceParseBackingJSONVxHS, 0},
|
|
||||||
@@ -4503,6 +4627,7 @@ virStorageSourceNetworkDefaultPort(virStorageNetProtocol protocol)
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
return 24007;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
/* we don't provide a default for RBD */
|
|
||||||
return 0;
|
|
||||||
diff --git a/src/util/virstoragefile.h b/src/util/virstoragefile.h
|
|
||||||
index 5689c39..3eb4e3c 100644
|
|
||||||
--- a/src/util/virstoragefile.h
|
|
||||||
+++ b/src/util/virstoragefile.h
|
|
||||||
@@ -136,6 +136,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_SSH,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_VXHS,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_NFS,
|
|
||||||
+ VIR_STORAGE_NET_PROTOCOL_VITASTOR,
|
|
||||||
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_LAST
|
|
||||||
} virStorageNetProtocol;
|
|
||||||
diff --git a/tests/storagepoolcapsschemadata/poolcaps-fs.xml b/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
index eee75af..8bd0a57 100644
|
|
||||||
--- a/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
+++ b/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
@@ -204,4 +204,11 @@
|
|
||||||
</enum>
|
|
||||||
</volOptions>
|
|
||||||
</pool>
|
|
||||||
+ <pool type='vitastor' supported='no'>
|
|
||||||
+ <volOptions>
|
|
||||||
+ <defaultFormat type='raw'/>
|
|
||||||
+ <enum name='targetFormatType'>
|
|
||||||
+ </enum>
|
|
||||||
+ </volOptions>
|
|
||||||
+ </pool>
|
|
||||||
</storagepoolCapabilities>
|
|
||||||
diff --git a/tests/storagepoolcapsschemadata/poolcaps-full.xml b/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
index 805950a..852df0d 100644
|
|
||||||
--- a/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
+++ b/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
@@ -204,4 +204,11 @@
|
|
||||||
</enum>
|
|
||||||
</volOptions>
|
|
||||||
</pool>
|
|
||||||
+ <pool type='vitastor' supported='yes'>
|
|
||||||
+ <volOptions>
|
|
||||||
+ <defaultFormat type='raw'/>
|
|
||||||
+ <enum name='targetFormatType'>
|
|
||||||
+ </enum>
|
|
||||||
+ </volOptions>
|
|
||||||
+ </pool>
|
|
||||||
</storagepoolCapabilities>
|
|
||||||
diff --git a/tests/storagepoolxml2argvtest.c b/tests/storagepoolxml2argvtest.c
|
|
||||||
index 967d1f2..1e8ff7a 100644
|
|
||||||
--- a/tests/storagepoolxml2argvtest.c
|
|
||||||
+++ b/tests/storagepoolxml2argvtest.c
|
|
||||||
@@ -68,6 +68,7 @@ testCompareXMLToArgvFiles(bool shouldFail,
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_ZFS:
|
|
||||||
case VIR_STORAGE_POOL_VSTORAGE:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
default:
|
|
||||||
VIR_TEST_DEBUG("pool type '%s' has no xml2argv test", defTypeStr);
|
|
||||||
diff --git a/tools/virsh-pool.c b/tools/virsh-pool.c
|
|
||||||
index 7835fa6..8841fcf 100644
|
|
||||||
--- a/tools/virsh-pool.c
|
|
||||||
+++ b/tools/virsh-pool.c
|
|
||||||
@@ -1237,6 +1237,9 @@ cmdPoolList(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED)
|
|
||||||
case VIR_STORAGE_POOL_VSTORAGE:
|
|
||||||
flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE;
|
|
||||||
break;
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
+ flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR;
|
|
||||||
+ break;
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
}
|
|
|
@ -1,692 +0,0 @@
|
||||||
commit c97d7f2bfb7798f0d68bdba2646245dcfb940efa
|
|
||||||
Author: Vitaliy Filippov <vitalif@yourcmc.ru>
|
|
||||||
Date: Mon Jun 28 01:20:19 2021 +0300
|
|
||||||
|
|
||||||
Add Vitastor support
|
|
||||||
|
|
||||||
diff --git a/docs/schemas/domaincommon.rng b/docs/schemas/domaincommon.rng
|
|
||||||
index 5ea14b6..a9df168 100644
|
|
||||||
--- a/docs/schemas/domaincommon.rng
|
|
||||||
+++ b/docs/schemas/domaincommon.rng
|
|
||||||
@@ -1859,6 +1859,35 @@
|
|
||||||
</element>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
+ <define name="diskSourceNetworkProtocolVitastor">
|
|
||||||
+ <element name="source">
|
|
||||||
+ <interleave>
|
|
||||||
+ <attribute name="protocol">
|
|
||||||
+ <value>vitastor</value>
|
|
||||||
+ </attribute>
|
|
||||||
+ <ref name="diskSourceCommon"/>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="name"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="query"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <zeroOrMore>
|
|
||||||
+ <ref name="diskSourceNetworkHost"/>
|
|
||||||
+ </zeroOrMore>
|
|
||||||
+ <optional>
|
|
||||||
+ <element name="config">
|
|
||||||
+ <attribute name="file">
|
|
||||||
+ <ref name="absFilePath"/>
|
|
||||||
+ </attribute>
|
|
||||||
+ <empty/>
|
|
||||||
+ </element>
|
|
||||||
+ </optional>
|
|
||||||
+ <empty/>
|
|
||||||
+ </interleave>
|
|
||||||
+ </element>
|
|
||||||
+ </define>
|
|
||||||
+
|
|
||||||
<define name="diskSourceNetworkProtocolISCSI">
|
|
||||||
<element name="source">
|
|
||||||
<attribute name="protocol">
|
|
||||||
@@ -2115,6 +2144,7 @@
|
|
||||||
<ref name="diskSourceNetworkProtocolSimple"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolVxHS"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolNFS"/>
|
|
||||||
+ <ref name="diskSourceNetworkProtocolVitastor"/>
|
|
||||||
</choice>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
diff --git a/include/libvirt/libvirt-storage.h b/include/libvirt/libvirt-storage.h
|
|
||||||
index 089e1e0..d7e7ef4 100644
|
|
||||||
--- a/include/libvirt/libvirt-storage.h
|
|
||||||
+++ b/include/libvirt/libvirt-storage.h
|
|
||||||
@@ -245,6 +245,7 @@ typedef enum {
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS = 1 << 17,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE = 1 << 18,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ISCSI_DIRECT = 1 << 19,
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR = 1 << 20,
|
|
||||||
} virConnectListAllStoragePoolsFlags;
|
|
||||||
|
|
||||||
int virConnectListAllStoragePools(virConnectPtr conn,
|
|
||||||
diff --git a/src/conf/domain_conf.c b/src/conf/domain_conf.c
|
|
||||||
index d78f846..f7222e3 100644
|
|
||||||
--- a/src/conf/domain_conf.c
|
|
||||||
+++ b/src/conf/domain_conf.c
|
|
||||||
@@ -8251,7 +8251,8 @@ virDomainDiskSourceNetworkParse(xmlNodePtr node,
|
|
||||||
src->configFile = virXPathString("string(./config/@file)", ctxt);
|
|
||||||
|
|
||||||
if (src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTP ||
|
|
||||||
- src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTPS)
|
|
||||||
+ src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTPS ||
|
|
||||||
+ src->protocol == VIR_STORAGE_NET_PROTOCOL_VITASTOR)
|
|
||||||
src->query = virXMLPropString(node, "query");
|
|
||||||
|
|
||||||
if (virDomainStorageNetworkParseHosts(node, ctxt, &src->hosts, &src->nhosts) < 0)
|
|
||||||
@@ -30775,6 +30776,7 @@ virDomainStorageSourceTranslateSourcePool(virStorageSource *src,
|
|
||||||
|
|
||||||
case VIR_STORAGE_POOL_MPATH:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
diff --git a/src/conf/domain_validate.c b/src/conf/domain_validate.c
|
|
||||||
index 2124d25..6acc6fa 100644
|
|
||||||
--- a/src/conf/domain_validate.c
|
|
||||||
+++ b/src/conf/domain_validate.c
|
|
||||||
@@ -470,7 +470,7 @@ virDomainDiskDefValidateSourceChainOne(const virStorageSource *src)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- /* internal snapshots and config files are currently supported only with rbd: */
|
|
||||||
+ /* internal snapshots are currently supported only with rbd: */
|
|
||||||
if (virStorageSourceGetActualType(src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD) {
|
|
||||||
if (src->snapshot) {
|
|
||||||
@@ -479,11 +479,15 @@ virDomainDiskDefValidateSourceChainOne(const virStorageSource *src)
|
|
||||||
"only with 'rbd' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
-
|
|
||||||
+ }
|
|
||||||
+ /* config files are currently supported only with rbd and vitastor: */
|
|
||||||
+ if (virStorageSourceGetActualType(src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD &&
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_VITASTOR) {
|
|
||||||
if (src->configFile) {
|
|
||||||
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
_("<config> element is currently supported "
|
|
||||||
- "only with 'rbd' disks"));
|
|
||||||
+ "only with 'rbd' and 'vitastor' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
diff --git a/src/conf/storage_conf.c b/src/conf/storage_conf.c
|
|
||||||
index 2aa9a3d..166ca1f 100644
|
|
||||||
--- a/src/conf/storage_conf.c
|
|
||||||
+++ b/src/conf/storage_conf.c
|
|
||||||
@@ -60,7 +60,7 @@ VIR_ENUM_IMPL(virStoragePool,
|
|
||||||
"logical", "disk", "iscsi",
|
|
||||||
"iscsi-direct", "scsi", "mpath",
|
|
||||||
"rbd", "sheepdog", "gluster",
|
|
||||||
- "zfs", "vstorage",
|
|
||||||
+ "zfs", "vstorage", "vitastor",
|
|
||||||
);
|
|
||||||
|
|
||||||
VIR_ENUM_IMPL(virStoragePoolFormatFileSystem,
|
|
||||||
@@ -246,6 +246,18 @@ static virStoragePoolTypeInfo poolTypeInfo[] = {
|
|
||||||
.formatToString = virStorageFileFormatTypeToString,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
+ {.poolType = VIR_STORAGE_POOL_VITASTOR,
|
|
||||||
+ .poolOptions = {
|
|
||||||
+ .flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NETWORK |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NAME),
|
|
||||||
+ },
|
|
||||||
+ .volOptions = {
|
|
||||||
+ .defaultFormat = VIR_STORAGE_FILE_RAW,
|
|
||||||
+ .formatFromString = virStorageVolumeFormatFromString,
|
|
||||||
+ .formatToString = virStorageFileFormatTypeToString,
|
|
||||||
+ }
|
|
||||||
+ },
|
|
||||||
{.poolType = VIR_STORAGE_POOL_SHEEPDOG,
|
|
||||||
.poolOptions = {
|
|
||||||
.flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
@@ -546,6 +558,11 @@ virStoragePoolDefParseSource(xmlXPathContextPtr ctxt,
|
|
||||||
_("element 'name' is mandatory for RBD pool"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
+ if (pool_type == VIR_STORAGE_POOL_VITASTOR && source->name == NULL) {
|
|
||||||
+ virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
+ _("element 'name' is mandatory for Vitastor pool"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
|
|
||||||
if (options->formatFromString) {
|
|
||||||
g_autofree char *format = NULL;
|
|
||||||
@@ -1182,6 +1199,7 @@ virStoragePoolDefFormatBuf(virBuffer *buf,
|
|
||||||
/* RBD, Sheepdog, Gluster and Iscsi-direct devices are not local block devs nor
|
|
||||||
* files, so they don't have a target */
|
|
||||||
if (def->type != VIR_STORAGE_POOL_RBD &&
|
|
||||||
+ def->type != VIR_STORAGE_POOL_VITASTOR &&
|
|
||||||
def->type != VIR_STORAGE_POOL_SHEEPDOG &&
|
|
||||||
def->type != VIR_STORAGE_POOL_GLUSTER &&
|
|
||||||
def->type != VIR_STORAGE_POOL_ISCSI_DIRECT) {
|
|
||||||
diff --git a/src/conf/storage_conf.h b/src/conf/storage_conf.h
|
|
||||||
index 76efaac..928149a 100644
|
|
||||||
--- a/src/conf/storage_conf.h
|
|
||||||
+++ b/src/conf/storage_conf.h
|
|
||||||
@@ -106,6 +106,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_POOL_GLUSTER, /* Gluster device */
|
|
||||||
VIR_STORAGE_POOL_ZFS, /* ZFS */
|
|
||||||
VIR_STORAGE_POOL_VSTORAGE, /* Virtuozzo Storage */
|
|
||||||
+ VIR_STORAGE_POOL_VITASTOR, /* Vitastor */
|
|
||||||
|
|
||||||
VIR_STORAGE_POOL_LAST,
|
|
||||||
} virStoragePoolType;
|
|
||||||
@@ -465,6 +466,7 @@ VIR_ENUM_DECL(virStoragePartedFs);
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SCSI | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_MPATH | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_RBD | \
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS | \
|
|
||||||
diff --git a/src/conf/storage_source_conf.c b/src/conf/storage_source_conf.c
|
|
||||||
index 5ca06fa..05ded49 100644
|
|
||||||
--- a/src/conf/storage_source_conf.c
|
|
||||||
+++ b/src/conf/storage_source_conf.c
|
|
||||||
@@ -85,6 +85,7 @@ VIR_ENUM_IMPL(virStorageNetProtocol,
|
|
||||||
"ssh",
|
|
||||||
"vxhs",
|
|
||||||
"nfs",
|
|
||||||
+ "vitastor",
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1262,6 +1263,7 @@ virStorageSourceNetworkDefaultPort(virStorageNetProtocol protocol)
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
return 24007;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
/* we don't provide a default for RBD */
|
|
||||||
return 0;
|
|
||||||
diff --git a/src/conf/storage_source_conf.h b/src/conf/storage_source_conf.h
|
|
||||||
index 389c7b5..dbf02e3 100644
|
|
||||||
--- a/src/conf/storage_source_conf.h
|
|
||||||
+++ b/src/conf/storage_source_conf.h
|
|
||||||
@@ -127,6 +127,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_SSH,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_VXHS,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_NFS,
|
|
||||||
+ VIR_STORAGE_NET_PROTOCOL_VITASTOR,
|
|
||||||
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_LAST
|
|
||||||
} virStorageNetProtocol;
|
|
||||||
diff --git a/src/conf/virstorageobj.c b/src/conf/virstorageobj.c
|
|
||||||
index 24957d6..4520a73 100644
|
|
||||||
--- a/src/conf/virstorageobj.c
|
|
||||||
+++ b/src/conf/virstorageobj.c
|
|
||||||
@@ -1487,6 +1487,7 @@ virStoragePoolObjSourceFindDuplicateCb(const void *payload,
|
|
||||||
return 1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
@@ -1986,6 +1987,8 @@ virStoragePoolObjMatch(virStoragePoolObj *obj,
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_MPATH)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_RBD) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_RBD)) ||
|
|
||||||
+ (MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR) &&
|
|
||||||
+ (obj->def->type == VIR_STORAGE_POOL_VITASTOR)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_SHEEPDOG)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER) &&
|
|
||||||
diff --git a/src/libvirt-storage.c b/src/libvirt-storage.c
|
|
||||||
index 2a7cdca..f756be1 100644
|
|
||||||
--- a/src/libvirt-storage.c
|
|
||||||
+++ b/src/libvirt-storage.c
|
|
||||||
@@ -92,6 +92,7 @@ virStoragePoolGetConnect(virStoragePoolPtr pool)
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SCSI
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_MPATH
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_RBD
|
|
||||||
+ * VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_ZFS
|
|
||||||
diff --git a/src/libxl/libxl_conf.c b/src/libxl/libxl_conf.c
|
|
||||||
index 56cb9ab..dfb31b9 100644
|
|
||||||
--- a/src/libxl/libxl_conf.c
|
|
||||||
+++ b/src/libxl/libxl_conf.c
|
|
||||||
@@ -972,6 +972,7 @@ libxlMakeNetworkDiskSrcStr(virStorageSource *src,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
diff --git a/src/libxl/xen_xl.c b/src/libxl/xen_xl.c
|
|
||||||
index c0905b0..c172378 100644
|
|
||||||
--- a/src/libxl/xen_xl.c
|
|
||||||
+++ b/src/libxl/xen_xl.c
|
|
||||||
@@ -1540,6 +1540,7 @@ xenFormatXLDiskSrcNet(virStorageSource *src)
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
diff --git a/src/qemu/qemu_block.c b/src/qemu/qemu_block.c
|
|
||||||
index 6627d04..f769d24 100644
|
|
||||||
--- a/src/qemu/qemu_block.c
|
|
||||||
+++ b/src/qemu/qemu_block.c
|
|
||||||
@@ -928,6 +928,38 @@ qemuBlockStorageSourceGetRBDProps(virStorageSource *src,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
+static virJSONValue *
|
|
||||||
+qemuBlockStorageSourceGetVitastorProps(virStorageSource *src)
|
|
||||||
+{
|
|
||||||
+ virJSONValuePtr ret = NULL;
|
|
||||||
+ virStorageNetHostDefPtr host;
|
|
||||||
+ size_t i;
|
|
||||||
+ g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
||||||
+ g_autofree char *etcd = NULL;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ host = src->hosts + i;
|
|
||||||
+ if ((virStorageNetHostTransport)host->transport != VIR_STORAGE_NET_HOST_TRANS_TCP) {
|
|
||||||
+ return NULL;
|
|
||||||
+ }
|
|
||||||
+ virBufferAsprintf(&buf, i > 0 ? ",%s:%u" : "%s:%u", host->name, host->port);
|
|
||||||
+ }
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ etcd = virBufferContentAndReset(&buf);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virJSONValueObjectCreate(&ret,
|
|
||||||
+ "S:etcd-host", etcd,
|
|
||||||
+ "S:etcd-prefix", src->query,
|
|
||||||
+ "S:config-path", src->configFile,
|
|
||||||
+ "s:image", src->path,
|
|
||||||
+ NULL) < 0)
|
|
||||||
+ return NULL;
|
|
||||||
+
|
|
||||||
+ return ret;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
static virJSONValue *
|
|
||||||
qemuBlockStorageSourceGetSheepdogProps(virStorageSource *src)
|
|
||||||
{
|
|
||||||
@@ -1218,6 +1250,12 @@ qemuBlockStorageSourceGetBackendProps(virStorageSource *src,
|
|
||||||
return NULL;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ driver = "vitastor";
|
|
||||||
+ if (!(fileprops = qemuBlockStorageSourceGetVitastorProps(src)))
|
|
||||||
+ return NULL;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
driver = "sheepdog";
|
|
||||||
if (!(fileprops = qemuBlockStorageSourceGetSheepdogProps(src)))
|
|
||||||
@@ -2231,6 +2269,7 @@ qemuBlockGetBackingStoreString(virStorageSource *src,
|
|
||||||
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
@@ -2608,6 +2647,12 @@ qemuBlockStorageSourceCreateGetStorageProps(virStorageSource *src,
|
|
||||||
return -1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ driver = "vitastor";
|
|
||||||
+ if (!(location = qemuBlockStorageSourceGetVitastorProps(src)))
|
|
||||||
+ return -1;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
driver = "sheepdog";
|
|
||||||
if (!(location = qemuBlockStorageSourceGetSheepdogProps(src)))
|
|
||||||
diff --git a/src/qemu/qemu_command.c b/src/qemu/qemu_command.c
|
|
||||||
index ea51369..d714597 100644
|
|
||||||
--- a/src/qemu/qemu_command.c
|
|
||||||
+++ b/src/qemu/qemu_command.c
|
|
||||||
@@ -1074,6 +1074,43 @@ qemuBuildNetworkDriveStr(virStorageSource *src,
|
|
||||||
ret = virBufferContentAndReset(&buf);
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (strchr(src->path, ':')) {
|
|
||||||
+ virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
||||||
+ _("':' not allowed in Vitastor source volume name '%s'"),
|
|
||||||
+ src->path);
|
|
||||||
+ return NULL;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ virBufferStrcat(&buf, "vitastor:image=", src->path, NULL);
|
|
||||||
+
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ virBufferAddLit(&buf, ":etcd-host=");
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ if (i)
|
|
||||||
+ virBufferAddLit(&buf, ",");
|
|
||||||
+
|
|
||||||
+ /* assume host containing : is ipv6 */
|
|
||||||
+ if (strchr(src->hosts[i].name, ':'))
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", "[%s]",
|
|
||||||
+ src->hosts[i].name);
|
|
||||||
+ else
|
|
||||||
+ virBufferAsprintf(&buf, "%s", src->hosts[i].name);
|
|
||||||
+
|
|
||||||
+ if (src->hosts[i].port)
|
|
||||||
+ virBufferAsprintf(&buf, "\\:%u", src->hosts[i].port);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (src->configFile)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":config-path=%s", src->configFile);
|
|
||||||
+
|
|
||||||
+ if (src->query)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":etcd-prefix=%s", src->query);
|
|
||||||
+
|
|
||||||
+ ret = virBufferContentAndReset(&buf);
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
||||||
_("VxHS protocol does not support URI syntax"));
|
|
||||||
diff --git a/src/qemu/qemu_domain.c b/src/qemu/qemu_domain.c
|
|
||||||
index fc60e15..5ab410d 100644
|
|
||||||
--- a/src/qemu/qemu_domain.c
|
|
||||||
+++ b/src/qemu/qemu_domain.c
|
|
||||||
@@ -4829,7 +4829,8 @@ qemuDomainValidateStorageSource(virStorageSource *src,
|
|
||||||
if (src->query &&
|
|
||||||
(actualType != VIR_STORAGE_TYPE_NETWORK ||
|
|
||||||
(src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTPS &&
|
|
||||||
- src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTP))) {
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTP &&
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_VITASTOR))) {
|
|
||||||
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
||||||
_("query is supported only with HTTP(S) protocols"));
|
|
||||||
return -1;
|
|
||||||
@@ -10027,6 +10028,7 @@ qemuDomainPrepareStorageSourceTLS(virStorageSource *src,
|
|
||||||
break;
|
|
||||||
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
diff --git a/src/qemu/qemu_snapshot.c b/src/qemu/qemu_snapshot.c
|
|
||||||
index 4e74ddd..14e5f2e 100644
|
|
||||||
--- a/src/qemu/qemu_snapshot.c
|
|
||||||
+++ b/src/qemu/qemu_snapshot.c
|
|
||||||
@@ -402,6 +402,7 @@ qemuSnapshotPrepareDiskExternalInactive(virDomainSnapshotDiskDef *snapdisk,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
@@ -494,6 +495,7 @@ qemuSnapshotPrepareDiskExternalActive(virDomainObj *vm,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_HTTP:
|
|
||||||
@@ -647,6 +649,7 @@ qemuSnapshotPrepareDiskInternal(virDomainDiskDef *disk,
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
diff --git a/src/storage/storage_driver.c b/src/storage/storage_driver.c
|
|
||||||
index c2ff4b8..70d0689 100644
|
|
||||||
--- a/src/storage/storage_driver.c
|
|
||||||
+++ b/src/storage/storage_driver.c
|
|
||||||
@@ -1644,6 +1644,7 @@ storageVolLookupByPathCallback(virStoragePoolObj *obj,
|
|
||||||
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_ZFS:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
diff --git a/src/storage_file/storage_source_backingstore.c b/src/storage_file/storage_source_backingstore.c
|
|
||||||
index e48ae72..2017ccc 100644
|
|
||||||
--- a/src/storage_file/storage_source_backingstore.c
|
|
||||||
+++ b/src/storage_file/storage_source_backingstore.c
|
|
||||||
@@ -284,6 +284,75 @@ virStorageSourceParseRBDColonString(const char *rbdstr,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
+static int
|
|
||||||
+virStorageSourceParseVitastorColonString(const char *colonstr,
|
|
||||||
+ virStorageSource *src)
|
|
||||||
+{
|
|
||||||
+ char *p, *e, *next;
|
|
||||||
+ g_autofree char *options = NULL;
|
|
||||||
+
|
|
||||||
+ /* optionally skip the "vitastor:" prefix if provided */
|
|
||||||
+ if (STRPREFIX(colonstr, "vitastor:"))
|
|
||||||
+ colonstr += strlen("vitastor:");
|
|
||||||
+
|
|
||||||
+ options = g_strdup(colonstr);
|
|
||||||
+
|
|
||||||
+ p = options;
|
|
||||||
+ while (*p) {
|
|
||||||
+ /* find : delimiter or end of string */
|
|
||||||
+ for (e = p; *e && *e != ':'; ++e) {
|
|
||||||
+ if (*e == '\\') {
|
|
||||||
+ e++;
|
|
||||||
+ if (*e == '\0')
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ if (*e == '\0') {
|
|
||||||
+ next = e; /* last kv pair */
|
|
||||||
+ } else {
|
|
||||||
+ next = e + 1;
|
|
||||||
+ *e = '\0';
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (STRPREFIX(p, "image=")) {
|
|
||||||
+ src->path = g_strdup(p + strlen("image="));
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-prefix=")) {
|
|
||||||
+ src->query = g_strdup(p + strlen("etcd-prefix="));
|
|
||||||
+ } else if (STRPREFIX(p, "config-path=")) {
|
|
||||||
+ src->configFile = g_strdup(p + strlen("config-path="));
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-host=")) {
|
|
||||||
+ char *h, *sep;
|
|
||||||
+
|
|
||||||
+ h = p + strlen("etcd-host=");
|
|
||||||
+ while (h < e) {
|
|
||||||
+ for (sep = h; sep < e; ++sep) {
|
|
||||||
+ if (*sep == '\\' && (sep[1] == ',' ||
|
|
||||||
+ sep[1] == ';' ||
|
|
||||||
+ sep[1] == ' ')) {
|
|
||||||
+ *sep = '\0';
|
|
||||||
+ sep += 2;
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virStorageSourceRBDAddHost(src, h) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+
|
|
||||||
+ h = sep;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ p = next;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (!src->path) {
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
static int
|
|
||||||
virStorageSourceParseNBDColonString(const char *nbdstr,
|
|
||||||
virStorageSource *src)
|
|
||||||
@@ -396,6 +465,11 @@ virStorageSourceParseBackingColon(virStorageSource *src,
|
|
||||||
return -1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (virStorageSourceParseVitastorColonString(path, src) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
@@ -984,6 +1058,54 @@ virStorageSourceParseBackingJSONRBD(virStorageSource *src,
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
+static int
|
|
||||||
+virStorageSourceParseBackingJSONVitastor(virStorageSource *src,
|
|
||||||
+ virJSONValue *json,
|
|
||||||
+ const char *jsonstr G_GNUC_UNUSED,
|
|
||||||
+ int opaque G_GNUC_UNUSED)
|
|
||||||
+{
|
|
||||||
+ const char *filename;
|
|
||||||
+ const char *image = virJSONValueObjectGetString(json, "image");
|
|
||||||
+ const char *conf = virJSONValueObjectGetString(json, "config-path");
|
|
||||||
+ const char *etcd_prefix = virJSONValueObjectGetString(json, "etcd-prefix");
|
|
||||||
+ virJSONValue *servers = virJSONValueObjectGetArray(json, "server");
|
|
||||||
+ size_t nservers;
|
|
||||||
+ size_t i;
|
|
||||||
+
|
|
||||||
+ src->type = VIR_STORAGE_TYPE_NETWORK;
|
|
||||||
+ src->protocol = VIR_STORAGE_NET_PROTOCOL_VITASTOR;
|
|
||||||
+
|
|
||||||
+ /* legacy syntax passed via 'filename' option */
|
|
||||||
+ if ((filename = virJSONValueObjectGetString(json, "filename")))
|
|
||||||
+ return virStorageSourceParseVitastorColonString(filename, src);
|
|
||||||
+
|
|
||||||
+ if (!image) {
|
|
||||||
+ virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
||||||
+ _("missing image name in Vitastor backing volume "
|
|
||||||
+ "JSON specification"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ src->path = g_strdup(image);
|
|
||||||
+ src->configFile = g_strdup(conf);
|
|
||||||
+ src->query = g_strdup(etcd_prefix);
|
|
||||||
+
|
|
||||||
+ if (servers) {
|
|
||||||
+ nservers = virJSONValueArraySize(servers);
|
|
||||||
+
|
|
||||||
+ src->hosts = g_new0(virStorageNetHostDef, nservers);
|
|
||||||
+ src->nhosts = nservers;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < nservers; i++) {
|
|
||||||
+ if (virStorageSourceParseBackingJSONInetSocketAddress(src->hosts + i,
|
|
||||||
+ virJSONValueArrayGet(servers, i)) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
static int
|
|
||||||
virStorageSourceParseBackingJSONRaw(virStorageSource *src,
|
|
||||||
virJSONValue *json,
|
|
||||||
@@ -1162,6 +1284,7 @@ static const struct virStorageSourceJSONDriverParser jsonParsers[] = {
|
|
||||||
{"sheepdog", false, virStorageSourceParseBackingJSONSheepdog, 0},
|
|
||||||
{"ssh", false, virStorageSourceParseBackingJSONSSH, 0},
|
|
||||||
{"rbd", false, virStorageSourceParseBackingJSONRBD, 0},
|
|
||||||
+ {"vitastor", false, virStorageSourceParseBackingJSONVitastor, 0},
|
|
||||||
{"raw", true, virStorageSourceParseBackingJSONRaw, 0},
|
|
||||||
{"nfs", false, virStorageSourceParseBackingJSONNFS, 0},
|
|
||||||
{"vxhs", false, virStorageSourceParseBackingJSONVxHS, 0},
|
|
||||||
diff --git a/src/test/test_driver.c b/src/test/test_driver.c
|
|
||||||
index ef0ddab..2173dc3 100644
|
|
||||||
--- a/src/test/test_driver.c
|
|
||||||
+++ b/src/test/test_driver.c
|
|
||||||
@@ -7131,6 +7131,7 @@ testStorageVolumeTypeForPool(int pooltype)
|
|
||||||
case VIR_STORAGE_POOL_ISCSI_DIRECT:
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
return VIR_STORAGE_VOL_NETWORK;
|
|
||||||
case VIR_STORAGE_POOL_LOGICAL:
|
|
||||||
case VIR_STORAGE_POOL_DISK:
|
|
||||||
diff --git a/tests/storagepoolcapsschemadata/poolcaps-fs.xml b/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
index eee75af..8bd0a57 100644
|
|
||||||
--- a/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
+++ b/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
@@ -204,4 +204,11 @@
|
|
||||||
</enum>
|
|
||||||
</volOptions>
|
|
||||||
</pool>
|
|
||||||
+ <pool type='vitastor' supported='no'>
|
|
||||||
+ <volOptions>
|
|
||||||
+ <defaultFormat type='raw'/>
|
|
||||||
+ <enum name='targetFormatType'>
|
|
||||||
+ </enum>
|
|
||||||
+ </volOptions>
|
|
||||||
+ </pool>
|
|
||||||
</storagepoolCapabilities>
|
|
||||||
diff --git a/tests/storagepoolcapsschemadata/poolcaps-full.xml b/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
index 805950a..852df0d 100644
|
|
||||||
--- a/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
+++ b/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
@@ -204,4 +204,11 @@
|
|
||||||
</enum>
|
|
||||||
</volOptions>
|
|
||||||
</pool>
|
|
||||||
+ <pool type='vitastor' supported='yes'>
|
|
||||||
+ <volOptions>
|
|
||||||
+ <defaultFormat type='raw'/>
|
|
||||||
+ <enum name='targetFormatType'>
|
|
||||||
+ </enum>
|
|
||||||
+ </volOptions>
|
|
||||||
+ </pool>
|
|
||||||
</storagepoolCapabilities>
|
|
||||||
diff --git a/tests/storagepoolxml2argvtest.c b/tests/storagepoolxml2argvtest.c
|
|
||||||
index 449b745..7f95cc8 100644
|
|
||||||
--- a/tests/storagepoolxml2argvtest.c
|
|
||||||
+++ b/tests/storagepoolxml2argvtest.c
|
|
||||||
@@ -68,6 +68,7 @@ testCompareXMLToArgvFiles(bool shouldFail,
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_ZFS:
|
|
||||||
case VIR_STORAGE_POOL_VSTORAGE:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
default:
|
|
||||||
VIR_TEST_DEBUG("pool type '%s' has no xml2argv test", defTypeStr);
|
|
||||||
diff --git a/tools/virsh-pool.c b/tools/virsh-pool.c
|
|
||||||
index 18f3839..c8e1436 100644
|
|
||||||
--- a/tools/virsh-pool.c
|
|
||||||
+++ b/tools/virsh-pool.c
|
|
||||||
@@ -1231,6 +1231,9 @@ cmdPoolList(vshControl *ctl, const vshCmd *cmd G_GNUC_UNUSED)
|
|
||||||
case VIR_STORAGE_POOL_VSTORAGE:
|
|
||||||
flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE;
|
|
||||||
break;
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
+ flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR;
|
|
||||||
+ break;
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
}
|
|
|
@ -1,692 +0,0 @@
|
||||||
commit c97d7f2bfb7798f0d68bdba2646245dcfb940efa
|
|
||||||
Author: Vitaliy Filippov <vitalif@yourcmc.ru>
|
|
||||||
Date: Mon Jun 28 01:20:19 2021 +0300
|
|
||||||
|
|
||||||
Add Vitastor support
|
|
||||||
|
|
||||||
Index: libvirt-7.6.0/docs/schemas/domaincommon.rng
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/docs/schemas/domaincommon.rng
|
|
||||||
+++ libvirt-7.6.0/docs/schemas/domaincommon.rng
|
|
||||||
@@ -1877,6 +1877,35 @@
|
|
||||||
</element>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
+ <define name="diskSourceNetworkProtocolVitastor">
|
|
||||||
+ <element name="source">
|
|
||||||
+ <interleave>
|
|
||||||
+ <attribute name="protocol">
|
|
||||||
+ <value>vitastor</value>
|
|
||||||
+ </attribute>
|
|
||||||
+ <ref name="diskSourceCommon"/>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="name"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <optional>
|
|
||||||
+ <attribute name="query"/>
|
|
||||||
+ </optional>
|
|
||||||
+ <zeroOrMore>
|
|
||||||
+ <ref name="diskSourceNetworkHost"/>
|
|
||||||
+ </zeroOrMore>
|
|
||||||
+ <optional>
|
|
||||||
+ <element name="config">
|
|
||||||
+ <attribute name="file">
|
|
||||||
+ <ref name="absFilePath"/>
|
|
||||||
+ </attribute>
|
|
||||||
+ <empty/>
|
|
||||||
+ </element>
|
|
||||||
+ </optional>
|
|
||||||
+ <empty/>
|
|
||||||
+ </interleave>
|
|
||||||
+ </element>
|
|
||||||
+ </define>
|
|
||||||
+
|
|
||||||
<define name="diskSourceNetworkProtocolISCSI">
|
|
||||||
<element name="source">
|
|
||||||
<attribute name="protocol">
|
|
||||||
@@ -2133,6 +2162,7 @@
|
|
||||||
<ref name="diskSourceNetworkProtocolSimple"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolVxHS"/>
|
|
||||||
<ref name="diskSourceNetworkProtocolNFS"/>
|
|
||||||
+ <ref name="diskSourceNetworkProtocolVitastor"/>
|
|
||||||
</choice>
|
|
||||||
</define>
|
|
||||||
|
|
||||||
Index: libvirt-7.6.0/include/libvirt/libvirt-storage.h
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/include/libvirt/libvirt-storage.h
|
|
||||||
+++ libvirt-7.6.0/include/libvirt/libvirt-storage.h
|
|
||||||
@@ -245,6 +245,7 @@ typedef enum {
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS = 1 << 17,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE = 1 << 18,
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ISCSI_DIRECT = 1 << 19,
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR = 1 << 20,
|
|
||||||
} virConnectListAllStoragePoolsFlags;
|
|
||||||
|
|
||||||
int virConnectListAllStoragePools(virConnectPtr conn,
|
|
||||||
Index: libvirt-7.6.0/src/conf/domain_conf.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/conf/domain_conf.c
|
|
||||||
+++ libvirt-7.6.0/src/conf/domain_conf.c
|
|
||||||
@@ -8268,7 +8268,8 @@ virDomainDiskSourceNetworkParse(xmlNodeP
|
|
||||||
src->configFile = virXPathString("string(./config/@file)", ctxt);
|
|
||||||
|
|
||||||
if (src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTP ||
|
|
||||||
- src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTPS)
|
|
||||||
+ src->protocol == VIR_STORAGE_NET_PROTOCOL_HTTPS ||
|
|
||||||
+ src->protocol == VIR_STORAGE_NET_PROTOCOL_VITASTOR)
|
|
||||||
src->query = virXMLPropString(node, "query");
|
|
||||||
|
|
||||||
if (virDomainStorageNetworkParseHosts(node, ctxt, &src->hosts, &src->nhosts) < 0)
|
|
||||||
@@ -30831,6 +30832,7 @@ virDomainStorageSourceTranslateSourcePoo
|
|
||||||
|
|
||||||
case VIR_STORAGE_POOL_MPATH:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
Index: libvirt-7.6.0/src/conf/domain_validate.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/conf/domain_validate.c
|
|
||||||
+++ libvirt-7.6.0/src/conf/domain_validate.c
|
|
||||||
@@ -470,7 +470,7 @@ virDomainDiskDefValidateSourceChainOne(c
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
- /* internal snapshots and config files are currently supported only with rbd: */
|
|
||||||
+ /* internal snapshots are currently supported only with rbd: */
|
|
||||||
if (virStorageSourceGetActualType(src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD) {
|
|
||||||
if (src->snapshot) {
|
|
||||||
@@ -479,11 +479,15 @@ virDomainDiskDefValidateSourceChainOne(c
|
|
||||||
"only with 'rbd' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
-
|
|
||||||
+ }
|
|
||||||
+ /* config files are currently supported only with rbd and vitastor: */
|
|
||||||
+ if (virStorageSourceGetActualType(src) != VIR_STORAGE_TYPE_NETWORK &&
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_RBD &&
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_VITASTOR) {
|
|
||||||
if (src->configFile) {
|
|
||||||
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
_("<config> element is currently supported "
|
|
||||||
- "only with 'rbd' disks"));
|
|
||||||
+ "only with 'rbd' and 'vitastor' disks"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Index: libvirt-7.6.0/src/conf/storage_conf.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/conf/storage_conf.c
|
|
||||||
+++ libvirt-7.6.0/src/conf/storage_conf.c
|
|
||||||
@@ -60,7 +60,7 @@ VIR_ENUM_IMPL(virStoragePool,
|
|
||||||
"logical", "disk", "iscsi",
|
|
||||||
"iscsi-direct", "scsi", "mpath",
|
|
||||||
"rbd", "sheepdog", "gluster",
|
|
||||||
- "zfs", "vstorage",
|
|
||||||
+ "zfs", "vstorage", "vitastor",
|
|
||||||
);
|
|
||||||
|
|
||||||
VIR_ENUM_IMPL(virStoragePoolFormatFileSystem,
|
|
||||||
@@ -246,6 +246,18 @@ static virStoragePoolTypeInfo poolTypeIn
|
|
||||||
.formatToString = virStorageFileFormatTypeToString,
|
|
||||||
}
|
|
||||||
},
|
|
||||||
+ {.poolType = VIR_STORAGE_POOL_VITASTOR,
|
|
||||||
+ .poolOptions = {
|
|
||||||
+ .flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NETWORK |
|
|
||||||
+ VIR_STORAGE_POOL_SOURCE_NAME),
|
|
||||||
+ },
|
|
||||||
+ .volOptions = {
|
|
||||||
+ .defaultFormat = VIR_STORAGE_FILE_RAW,
|
|
||||||
+ .formatFromString = virStorageVolumeFormatFromString,
|
|
||||||
+ .formatToString = virStorageFileFormatTypeToString,
|
|
||||||
+ }
|
|
||||||
+ },
|
|
||||||
{.poolType = VIR_STORAGE_POOL_SHEEPDOG,
|
|
||||||
.poolOptions = {
|
|
||||||
.flags = (VIR_STORAGE_POOL_SOURCE_HOST |
|
|
||||||
@@ -546,6 +558,11 @@ virStoragePoolDefParseSource(xmlXPathCon
|
|
||||||
_("element 'name' is mandatory for RBD pool"));
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
+ if (pool_type == VIR_STORAGE_POOL_VITASTOR && source->name == NULL) {
|
|
||||||
+ virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
||||||
+ _("element 'name' is mandatory for Vitastor pool"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
|
|
||||||
if (options->formatFromString) {
|
|
||||||
g_autofree char *format = NULL;
|
|
||||||
@@ -1182,6 +1199,7 @@ virStoragePoolDefFormatBuf(virBuffer *bu
|
|
||||||
/* RBD, Sheepdog, Gluster and Iscsi-direct devices are not local block devs nor
|
|
||||||
* files, so they don't have a target */
|
|
||||||
if (def->type != VIR_STORAGE_POOL_RBD &&
|
|
||||||
+ def->type != VIR_STORAGE_POOL_VITASTOR &&
|
|
||||||
def->type != VIR_STORAGE_POOL_SHEEPDOG &&
|
|
||||||
def->type != VIR_STORAGE_POOL_GLUSTER &&
|
|
||||||
def->type != VIR_STORAGE_POOL_ISCSI_DIRECT) {
|
|
||||||
Index: libvirt-7.6.0/src/conf/storage_conf.h
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/conf/storage_conf.h
|
|
||||||
+++ libvirt-7.6.0/src/conf/storage_conf.h
|
|
||||||
@@ -106,6 +106,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_POOL_GLUSTER, /* Gluster device */
|
|
||||||
VIR_STORAGE_POOL_ZFS, /* ZFS */
|
|
||||||
VIR_STORAGE_POOL_VSTORAGE, /* Virtuozzo Storage */
|
|
||||||
+ VIR_STORAGE_POOL_VITASTOR, /* Vitastor */
|
|
||||||
|
|
||||||
VIR_STORAGE_POOL_LAST,
|
|
||||||
} virStoragePoolType;
|
|
||||||
@@ -465,6 +466,7 @@ VIR_ENUM_DECL(virStoragePartedFs);
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SCSI | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_MPATH | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_RBD | \
|
|
||||||
+ VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER | \
|
|
||||||
VIR_CONNECT_LIST_STORAGE_POOLS_ZFS | \
|
|
||||||
Index: libvirt-7.6.0/src/conf/storage_source_conf.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/conf/storage_source_conf.c
|
|
||||||
+++ libvirt-7.6.0/src/conf/storage_source_conf.c
|
|
||||||
@@ -85,6 +85,7 @@ VIR_ENUM_IMPL(virStorageNetProtocol,
|
|
||||||
"ssh",
|
|
||||||
"vxhs",
|
|
||||||
"nfs",
|
|
||||||
+ "vitastor",
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1262,6 +1263,7 @@ virStorageSourceNetworkDefaultPort(virSt
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
return 24007;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
/* we don't provide a default for RBD */
|
|
||||||
return 0;
|
|
||||||
Index: libvirt-7.6.0/src/conf/storage_source_conf.h
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/conf/storage_source_conf.h
|
|
||||||
+++ libvirt-7.6.0/src/conf/storage_source_conf.h
|
|
||||||
@@ -127,6 +127,7 @@ typedef enum {
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_SSH,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_VXHS,
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_NFS,
|
|
||||||
+ VIR_STORAGE_NET_PROTOCOL_VITASTOR,
|
|
||||||
|
|
||||||
VIR_STORAGE_NET_PROTOCOL_LAST
|
|
||||||
} virStorageNetProtocol;
|
|
||||||
Index: libvirt-7.6.0/src/conf/virstorageobj.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/conf/virstorageobj.c
|
|
||||||
+++ libvirt-7.6.0/src/conf/virstorageobj.c
|
|
||||||
@@ -1481,6 +1481,7 @@ virStoragePoolObjSourceFindDuplicateCb(c
|
|
||||||
return 1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
@@ -1980,6 +1981,8 @@ virStoragePoolObjMatch(virStoragePoolObj
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_MPATH)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_RBD) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_RBD)) ||
|
|
||||||
+ (MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR) &&
|
|
||||||
+ (obj->def->type == VIR_STORAGE_POOL_VITASTOR)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG) &&
|
|
||||||
(obj->def->type == VIR_STORAGE_POOL_SHEEPDOG)) ||
|
|
||||||
(MATCH(VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER) &&
|
|
||||||
Index: libvirt-7.6.0/src/libvirt-storage.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/libvirt-storage.c
|
|
||||||
+++ libvirt-7.6.0/src/libvirt-storage.c
|
|
||||||
@@ -92,6 +92,7 @@ virStoragePoolGetConnect(virStoragePoolP
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SCSI
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_MPATH
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_RBD
|
|
||||||
+ * VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_SHEEPDOG
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_GLUSTER
|
|
||||||
* VIR_CONNECT_LIST_STORAGE_POOLS_ZFS
|
|
||||||
Index: libvirt-7.6.0/src/libxl/libxl_conf.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/libxl/libxl_conf.c
|
|
||||||
+++ libvirt-7.6.0/src/libxl/libxl_conf.c
|
|
||||||
@@ -972,6 +972,7 @@ libxlMakeNetworkDiskSrcStr(virStorageSou
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
Index: libvirt-7.6.0/src/libxl/xen_xl.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/libxl/xen_xl.c
|
|
||||||
+++ libvirt-7.6.0/src/libxl/xen_xl.c
|
|
||||||
@@ -1540,6 +1540,7 @@ xenFormatXLDiskSrcNet(virStorageSource *
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
virReportError(VIR_ERR_NO_SUPPORT,
|
|
||||||
Index: libvirt-7.6.0/src/qemu/qemu_block.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/qemu/qemu_block.c
|
|
||||||
+++ libvirt-7.6.0/src/qemu/qemu_block.c
|
|
||||||
@@ -916,6 +916,38 @@ qemuBlockStorageSourceGetRBDProps(virSto
|
|
||||||
|
|
||||||
|
|
||||||
static virJSONValue *
|
|
||||||
+qemuBlockStorageSourceGetVitastorProps(virStorageSource *src)
|
|
||||||
+{
|
|
||||||
+ virJSONValue *ret = NULL;
|
|
||||||
+ virStorageNetHostDef *host;
|
|
||||||
+ size_t i;
|
|
||||||
+ g_auto(virBuffer) buf = VIR_BUFFER_INITIALIZER;
|
|
||||||
+ g_autofree char *etcd = NULL;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ host = src->hosts + i;
|
|
||||||
+ if ((virStorageNetHostTransport)host->transport != VIR_STORAGE_NET_HOST_TRANS_TCP) {
|
|
||||||
+ return NULL;
|
|
||||||
+ }
|
|
||||||
+ virBufferAsprintf(&buf, i > 0 ? ",%s:%u" : "%s:%u", host->name, host->port);
|
|
||||||
+ }
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ etcd = virBufferContentAndReset(&buf);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virJSONValueObjectCreate(&ret,
|
|
||||||
+ "S:etcd-host", etcd,
|
|
||||||
+ "S:etcd-prefix", src->query,
|
|
||||||
+ "S:config-path", src->configFile,
|
|
||||||
+ "s:image", src->path,
|
|
||||||
+ NULL) < 0)
|
|
||||||
+ return NULL;
|
|
||||||
+
|
|
||||||
+ return ret;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+static virJSONValue *
|
|
||||||
qemuBlockStorageSourceGetSheepdogProps(virStorageSource *src)
|
|
||||||
{
|
|
||||||
g_autoptr(virJSONValue) serverprops = NULL;
|
|
||||||
@@ -1205,6 +1237,12 @@ qemuBlockStorageSourceGetBackendProps(vi
|
|
||||||
return NULL;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ driver = "vitastor";
|
|
||||||
+ if (!(fileprops = qemuBlockStorageSourceGetVitastorProps(src)))
|
|
||||||
+ return NULL;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
driver = "sheepdog";
|
|
||||||
if (!(fileprops = qemuBlockStorageSourceGetSheepdogProps(src)))
|
|
||||||
@@ -2219,6 +2257,7 @@ qemuBlockGetBackingStoreString(virStorag
|
|
||||||
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NFS:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SSH:
|
|
||||||
@@ -2596,6 +2635,12 @@ qemuBlockStorageSourceCreateGetStoragePr
|
|
||||||
return -1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ driver = "vitastor";
|
|
||||||
+ if (!(location = qemuBlockStorageSourceGetVitastorProps(src)))
|
|
||||||
+ return -1;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
driver = "sheepdog";
|
|
||||||
if (!(location = qemuBlockStorageSourceGetSheepdogProps(src)))
|
|
||||||
Index: libvirt-7.6.0/src/qemu/qemu_command.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/qemu/qemu_command.c
|
|
||||||
+++ libvirt-7.6.0/src/qemu/qemu_command.c
|
|
||||||
@@ -1074,6 +1074,43 @@ qemuBuildNetworkDriveStr(virStorageSourc
|
|
||||||
ret = virBufferContentAndReset(&buf);
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (strchr(src->path, ':')) {
|
|
||||||
+ virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
||||||
+ _("':' not allowed in Vitastor source volume name '%s'"),
|
|
||||||
+ src->path);
|
|
||||||
+ return NULL;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ virBufferStrcat(&buf, "vitastor:image=", src->path, NULL);
|
|
||||||
+
|
|
||||||
+ if (src->nhosts > 0) {
|
|
||||||
+ virBufferAddLit(&buf, ":etcd-host=");
|
|
||||||
+ for (i = 0; i < src->nhosts; i++) {
|
|
||||||
+ if (i)
|
|
||||||
+ virBufferAddLit(&buf, ",");
|
|
||||||
+
|
|
||||||
+ /* assume host containing : is ipv6 */
|
|
||||||
+ if (strchr(src->hosts[i].name, ':'))
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", "[%s]",
|
|
||||||
+ src->hosts[i].name);
|
|
||||||
+ else
|
|
||||||
+ virBufferAsprintf(&buf, "%s", src->hosts[i].name);
|
|
||||||
+
|
|
||||||
+ if (src->hosts[i].port)
|
|
||||||
+ virBufferAsprintf(&buf, "\\:%u", src->hosts[i].port);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (src->configFile)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":config-path=%s", src->configFile);
|
|
||||||
+
|
|
||||||
+ if (src->query)
|
|
||||||
+ virBufferEscape(&buf, '\\', ":", ":etcd-prefix=%s", src->query);
|
|
||||||
+
|
|
||||||
+ ret = virBufferContentAndReset(&buf);
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_VXHS:
|
|
||||||
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
||||||
_("VxHS protocol does not support URI syntax"));
|
|
||||||
Index: libvirt-7.6.0/src/qemu/qemu_domain.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/qemu/qemu_domain.c
|
|
||||||
+++ libvirt-7.6.0/src/qemu/qemu_domain.c
|
|
||||||
@@ -4900,7 +4900,8 @@ qemuDomainValidateStorageSource(virStora
|
|
||||||
if (src->query &&
|
|
||||||
(actualType != VIR_STORAGE_TYPE_NETWORK ||
|
|
||||||
(src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTPS &&
|
|
||||||
- src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTP))) {
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_HTTP &&
|
|
||||||
+ src->protocol != VIR_STORAGE_NET_PROTOCOL_VITASTOR))) {
|
|
||||||
virReportError(VIR_ERR_CONFIG_UNSUPPORTED, "%s",
|
|
||||||
_("query is supported only with HTTP(S) protocols"));
|
|
||||||
return -1;
|
|
||||||
@@ -10102,6 +10103,7 @@ qemuDomainPrepareStorageSourceTLS(virSto
|
|
||||||
break;
|
|
||||||
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
Index: libvirt-7.6.0/src/qemu/qemu_snapshot.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/qemu/qemu_snapshot.c
|
|
||||||
+++ libvirt-7.6.0/src/qemu/qemu_snapshot.c
|
|
||||||
@@ -402,6 +402,7 @@ qemuSnapshotPrepareDiskExternalInactive(
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
@@ -494,6 +495,7 @@ qemuSnapshotPrepareDiskExternalActive(vi
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_HTTP:
|
|
||||||
@@ -647,6 +649,7 @@ qemuSnapshotPrepareDiskInternal(virDomai
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
||||||
Index: libvirt-7.6.0/src/storage/storage_driver.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/storage/storage_driver.c
|
|
||||||
+++ libvirt-7.6.0/src/storage/storage_driver.c
|
|
||||||
@@ -1644,6 +1644,7 @@ storageVolLookupByPathCallback(virStorag
|
|
||||||
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_POOL_ZFS:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
Index: libvirt-7.6.0/src/storage_file/storage_source_backingstore.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/storage_file/storage_source_backingstore.c
|
|
||||||
+++ libvirt-7.6.0/src/storage_file/storage_source_backingstore.c
|
|
||||||
@@ -285,6 +285,75 @@ virStorageSourceParseRBDColonString(cons
|
|
||||||
|
|
||||||
|
|
||||||
static int
|
|
||||||
+virStorageSourceParseVitastorColonString(const char *colonstr,
|
|
||||||
+ virStorageSource *src)
|
|
||||||
+{
|
|
||||||
+ char *p, *e, *next;
|
|
||||||
+ g_autofree char *options = NULL;
|
|
||||||
+
|
|
||||||
+ /* optionally skip the "vitastor:" prefix if provided */
|
|
||||||
+ if (STRPREFIX(colonstr, "vitastor:"))
|
|
||||||
+ colonstr += strlen("vitastor:");
|
|
||||||
+
|
|
||||||
+ options = g_strdup(colonstr);
|
|
||||||
+
|
|
||||||
+ p = options;
|
|
||||||
+ while (*p) {
|
|
||||||
+ /* find : delimiter or end of string */
|
|
||||||
+ for (e = p; *e && *e != ':'; ++e) {
|
|
||||||
+ if (*e == '\\') {
|
|
||||||
+ e++;
|
|
||||||
+ if (*e == '\0')
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ if (*e == '\0') {
|
|
||||||
+ next = e; /* last kv pair */
|
|
||||||
+ } else {
|
|
||||||
+ next = e + 1;
|
|
||||||
+ *e = '\0';
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (STRPREFIX(p, "image=")) {
|
|
||||||
+ src->path = g_strdup(p + strlen("image="));
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-prefix=")) {
|
|
||||||
+ src->query = g_strdup(p + strlen("etcd-prefix="));
|
|
||||||
+ } else if (STRPREFIX(p, "config-path=")) {
|
|
||||||
+ src->configFile = g_strdup(p + strlen("config-path="));
|
|
||||||
+ } else if (STRPREFIX(p, "etcd-host=")) {
|
|
||||||
+ char *h, *sep;
|
|
||||||
+
|
|
||||||
+ h = p + strlen("etcd-host=");
|
|
||||||
+ while (h < e) {
|
|
||||||
+ for (sep = h; sep < e; ++sep) {
|
|
||||||
+ if (*sep == '\\' && (sep[1] == ',' ||
|
|
||||||
+ sep[1] == ';' ||
|
|
||||||
+ sep[1] == ' ')) {
|
|
||||||
+ *sep = '\0';
|
|
||||||
+ sep += 2;
|
|
||||||
+ break;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (virStorageSourceRBDAddHost(src, h) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+
|
|
||||||
+ h = sep;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ p = next;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ if (!src->path) {
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+static int
|
|
||||||
virStorageSourceParseNBDColonString(const char *nbdstr,
|
|
||||||
virStorageSource *src)
|
|
||||||
{
|
|
||||||
@@ -396,6 +465,11 @@ virStorageSourceParseBackingColon(virSto
|
|
||||||
return -1;
|
|
||||||
break;
|
|
||||||
|
|
||||||
+ case VIR_STORAGE_NET_PROTOCOL_VITASTOR:
|
|
||||||
+ if (virStorageSourceParseVitastorColonString(path, src) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ break;
|
|
||||||
+
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
||||||
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
||||||
@@ -985,6 +1059,54 @@ virStorageSourceParseBackingJSONRBD(virS
|
|
||||||
}
|
|
||||||
|
|
||||||
static int
|
|
||||||
+virStorageSourceParseBackingJSONVitastor(virStorageSource *src,
|
|
||||||
+ virJSONValue *json,
|
|
||||||
+ const char *jsonstr G_GNUC_UNUSED,
|
|
||||||
+ int opaque G_GNUC_UNUSED)
|
|
||||||
+{
|
|
||||||
+ const char *filename;
|
|
||||||
+ const char *image = virJSONValueObjectGetString(json, "image");
|
|
||||||
+ const char *conf = virJSONValueObjectGetString(json, "config-path");
|
|
||||||
+ const char *etcd_prefix = virJSONValueObjectGetString(json, "etcd-prefix");
|
|
||||||
+ virJSONValue *servers = virJSONValueObjectGetArray(json, "server");
|
|
||||||
+ size_t nservers;
|
|
||||||
+ size_t i;
|
|
||||||
+
|
|
||||||
+ src->type = VIR_STORAGE_TYPE_NETWORK;
|
|
||||||
+ src->protocol = VIR_STORAGE_NET_PROTOCOL_VITASTOR;
|
|
||||||
+
|
|
||||||
+ /* legacy syntax passed via 'filename' option */
|
|
||||||
+ if ((filename = virJSONValueObjectGetString(json, "filename")))
|
|
||||||
+ return virStorageSourceParseVitastorColonString(filename, src);
|
|
||||||
+
|
|
||||||
+ if (!image) {
|
|
||||||
+ virReportError(VIR_ERR_INVALID_ARG, "%s",
|
|
||||||
+ _("missing image name in Vitastor backing volume "
|
|
||||||
+ "JSON specification"));
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ src->path = g_strdup(image);
|
|
||||||
+ src->configFile = g_strdup(conf);
|
|
||||||
+ src->query = g_strdup(etcd_prefix);
|
|
||||||
+
|
|
||||||
+ if (servers) {
|
|
||||||
+ nservers = virJSONValueArraySize(servers);
|
|
||||||
+
|
|
||||||
+ src->hosts = g_new0(virStorageNetHostDef, nservers);
|
|
||||||
+ src->nhosts = nservers;
|
|
||||||
+
|
|
||||||
+ for (i = 0; i < nservers; i++) {
|
|
||||||
+ if (virStorageSourceParseBackingJSONInetSocketAddress(src->hosts + i,
|
|
||||||
+ virJSONValueArrayGet(servers, i)) < 0)
|
|
||||||
+ return -1;
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+
|
|
||||||
+static int
|
|
||||||
virStorageSourceParseBackingJSONRaw(virStorageSource *src,
|
|
||||||
virJSONValue *json,
|
|
||||||
const char *jsonstr,
|
|
||||||
@@ -1162,6 +1284,7 @@ static const struct virStorageSourceJSON
|
|
||||||
{"sheepdog", false, virStorageSourceParseBackingJSONSheepdog, 0},
|
|
||||||
{"ssh", false, virStorageSourceParseBackingJSONSSH, 0},
|
|
||||||
{"rbd", false, virStorageSourceParseBackingJSONRBD, 0},
|
|
||||||
+ {"vitastor", false, virStorageSourceParseBackingJSONVitastor, 0},
|
|
||||||
{"raw", true, virStorageSourceParseBackingJSONRaw, 0},
|
|
||||||
{"nfs", false, virStorageSourceParseBackingJSONNFS, 0},
|
|
||||||
{"vxhs", false, virStorageSourceParseBackingJSONVxHS, 0},
|
|
||||||
Index: libvirt-7.6.0/src/test/test_driver.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/src/test/test_driver.c
|
|
||||||
+++ libvirt-7.6.0/src/test/test_driver.c
|
|
||||||
@@ -7193,6 +7193,7 @@ testStorageVolumeTypeForPool(int pooltyp
|
|
||||||
case VIR_STORAGE_POOL_ISCSI_DIRECT:
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_RBD:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
return VIR_STORAGE_VOL_NETWORK;
|
|
||||||
case VIR_STORAGE_POOL_LOGICAL:
|
|
||||||
case VIR_STORAGE_POOL_DISK:
|
|
||||||
Index: libvirt-7.6.0/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
+++ libvirt-7.6.0/tests/storagepoolcapsschemadata/poolcaps-fs.xml
|
|
||||||
@@ -204,4 +204,11 @@
|
|
||||||
</enum>
|
|
||||||
</volOptions>
|
|
||||||
</pool>
|
|
||||||
+ <pool type='vitastor' supported='no'>
|
|
||||||
+ <volOptions>
|
|
||||||
+ <defaultFormat type='raw'/>
|
|
||||||
+ <enum name='targetFormatType'>
|
|
||||||
+ </enum>
|
|
||||||
+ </volOptions>
|
|
||||||
+ </pool>
|
|
||||||
</storagepoolCapabilities>
|
|
||||||
Index: libvirt-7.6.0/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
+++ libvirt-7.6.0/tests/storagepoolcapsschemadata/poolcaps-full.xml
|
|
||||||
@@ -204,4 +204,11 @@
|
|
||||||
</enum>
|
|
||||||
</volOptions>
|
|
||||||
</pool>
|
|
||||||
+ <pool type='vitastor' supported='yes'>
|
|
||||||
+ <volOptions>
|
|
||||||
+ <defaultFormat type='raw'/>
|
|
||||||
+ <enum name='targetFormatType'>
|
|
||||||
+ </enum>
|
|
||||||
+ </volOptions>
|
|
||||||
+ </pool>
|
|
||||||
</storagepoolCapabilities>
|
|
||||||
Index: libvirt-7.6.0/tests/storagepoolxml2argvtest.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/tests/storagepoolxml2argvtest.c
|
|
||||||
+++ libvirt-7.6.0/tests/storagepoolxml2argvtest.c
|
|
||||||
@@ -68,6 +68,7 @@ testCompareXMLToArgvFiles(bool shouldFai
|
|
||||||
case VIR_STORAGE_POOL_GLUSTER:
|
|
||||||
case VIR_STORAGE_POOL_ZFS:
|
|
||||||
case VIR_STORAGE_POOL_VSTORAGE:
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
default:
|
|
||||||
VIR_TEST_DEBUG("pool type '%s' has no xml2argv test", defTypeStr);
|
|
||||||
Index: libvirt-7.6.0/tools/virsh-pool.c
|
|
||||||
===================================================================
|
|
||||||
--- libvirt-7.6.0.orig/tools/virsh-pool.c
|
|
||||||
+++ libvirt-7.6.0/tools/virsh-pool.c
|
|
||||||
@@ -1231,6 +1231,9 @@ cmdPoolList(vshControl *ctl, const vshCm
|
|
||||||
case VIR_STORAGE_POOL_VSTORAGE:
|
|
||||||
flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VSTORAGE;
|
|
||||||
break;
|
|
||||||
+ case VIR_STORAGE_POOL_VITASTOR:
|
|
||||||
+ flags |= VIR_CONNECT_LIST_STORAGE_POOLS_VITASTOR;
|
|
||||||
+ break;
|
|
||||||
case VIR_STORAGE_POOL_LAST:
|
|
||||||
break;
|
|
||||||
}
|
|
|
@ -1,33 +0,0 @@
|
||||||
<!-- Example libvirt VM configuration with Vitastor disk -->
|
|
||||||
<domain type='kvm'>
|
|
||||||
<name>debian9</name>
|
|
||||||
<uuid>96f277fb-fd9c-49da-bf21-a5cfd54eb162</uuid>
|
|
||||||
<memory unit="KiB">524288</memory>
|
|
||||||
<currentMemory>524288</currentMemory>
|
|
||||||
<vcpu>1</vcpu>
|
|
||||||
<os>
|
|
||||||
<type arch='x86_64'>hvm</type>
|
|
||||||
<boot dev='hd' />
|
|
||||||
</os>
|
|
||||||
<devices>
|
|
||||||
<emulator>/usr/bin/qemu-system-x86_64</emulator>
|
|
||||||
<disk type='network' device='disk'>
|
|
||||||
<target dev='vda' bus='virtio' />
|
|
||||||
<driver name='qemu' type='raw' />
|
|
||||||
<!-- name is Vitastor image name -->
|
|
||||||
<!-- query (optional) is Vitastor's etcd_prefix -->
|
|
||||||
<source protocol='vitastor' name='debian9' query='/vitastor'>
|
|
||||||
<!-- config (optional) is the path to Vitastor's configuration file -->
|
|
||||||
<config file='/etc/vitastor/vitastor.conf' />
|
|
||||||
<!-- hosts = etcd addresses -->
|
|
||||||
<host name='192.168.7.2' port='2379' />
|
|
||||||
</source>
|
|
||||||
<!-- required because Vitastor only supports 4k physical sectors -->
|
|
||||||
<blockio physical_block_size="4096" logical_block_size="512" />
|
|
||||||
</disk>
|
|
||||||
<interface type='network'>
|
|
||||||
<source network='default' />
|
|
||||||
</interface>
|
|
||||||
<graphics type='vnc' port='-1' />
|
|
||||||
</devices>
|
|
||||||
</domain>
|
|
|
@ -1,288 +0,0 @@
|
||||||
diff --git a/nova/virt/image/model.py b/nova/virt/image/model.py
|
|
||||||
index 971f7e9c07..ec3fca72cb 100644
|
|
||||||
--- a/nova/virt/image/model.py
|
|
||||||
+++ b/nova/virt/image/model.py
|
|
||||||
@@ -129,3 +129,22 @@ class RBDImage(Image):
|
|
||||||
self.user = user
|
|
||||||
self.password = password
|
|
||||||
self.servers = servers
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+class VitastorImage(Image):
|
|
||||||
+ """Class for images in a remote Vitastor cluster"""
|
|
||||||
+
|
|
||||||
+ def __init__(self, name, etcd_address = None, etcd_prefix = None, config_path = None):
|
|
||||||
+ """Create a new Vitastor image object
|
|
||||||
+
|
|
||||||
+ :param name: name of the image
|
|
||||||
+ :param etcd_address: etcd URL(s) (optional)
|
|
||||||
+ :param etcd_prefix: etcd prefix (optional)
|
|
||||||
+ :param config_path: path to the configuration (optional)
|
|
||||||
+ """
|
|
||||||
+ super(VitastorImage, self).__init__(FORMAT_RAW)
|
|
||||||
+
|
|
||||||
+ self.name = name
|
|
||||||
+ self.etcd_address = etcd_address
|
|
||||||
+ self.etcd_prefix = etcd_prefix
|
|
||||||
+ self.config_path = config_path
|
|
||||||
diff --git a/nova/virt/images.py b/nova/virt/images.py
|
|
||||||
index 5358f3766a..ebe3d6effb 100644
|
|
||||||
--- a/nova/virt/images.py
|
|
||||||
+++ b/nova/virt/images.py
|
|
||||||
@@ -41,7 +41,7 @@ IMAGE_API = glance.API()
|
|
||||||
|
|
||||||
def qemu_img_info(path, format=None):
|
|
||||||
"""Return an object containing the parsed output from qemu-img info."""
|
|
||||||
- if not os.path.exists(path) and not path.startswith('rbd:'):
|
|
||||||
+ if not os.path.exists(path) and not path.startswith('rbd:') and not path.startswith('vitastor:'):
|
|
||||||
raise exception.DiskNotFound(location=path)
|
|
||||||
|
|
||||||
info = nova.privsep.qemu.unprivileged_qemu_img_info(path, format=format)
|
|
||||||
@@ -50,7 +50,7 @@ def qemu_img_info(path, format=None):
|
|
||||||
|
|
||||||
def privileged_qemu_img_info(path, format=None, output_format='json'):
|
|
||||||
"""Return an object containing the parsed output from qemu-img info."""
|
|
||||||
- if not os.path.exists(path) and not path.startswith('rbd:'):
|
|
||||||
+ if not os.path.exists(path) and not path.startswith('rbd:') and not path.startswith('vitastor:'):
|
|
||||||
raise exception.DiskNotFound(location=path)
|
|
||||||
|
|
||||||
info = nova.privsep.qemu.privileged_qemu_img_info(path, format=format)
|
|
||||||
diff --git a/nova/virt/libvirt/config.py b/nova/virt/libvirt/config.py
|
|
||||||
index ea525648b3..d7aa798954 100644
|
|
||||||
--- a/nova/virt/libvirt/config.py
|
|
||||||
+++ b/nova/virt/libvirt/config.py
|
|
||||||
@@ -1005,6 +1005,8 @@ class LibvirtConfigGuestDisk(LibvirtConfigGuestDevice):
|
|
||||||
self.driver_iommu = False
|
|
||||||
self.source_path = None
|
|
||||||
self.source_protocol = None
|
|
||||||
+ self.source_query = None
|
|
||||||
+ self.source_config = None
|
|
||||||
self.source_name = None
|
|
||||||
self.source_hosts = []
|
|
||||||
self.source_ports = []
|
|
||||||
@@ -1133,6 +1135,10 @@ class LibvirtConfigGuestDisk(LibvirtConfigGuestDevice):
|
|
||||||
source = etree.Element("source", protocol=self.source_protocol)
|
|
||||||
if self.source_name is not None:
|
|
||||||
source.set('name', self.source_name)
|
|
||||||
+ if self.source_query is not None:
|
|
||||||
+ source.set('query', self.source_query)
|
|
||||||
+ if self.source_config is not None:
|
|
||||||
+ source.append(etree.Element('config', file=self.source_config))
|
|
||||||
hosts_info = zip(self.source_hosts, self.source_ports)
|
|
||||||
for name, port in hosts_info:
|
|
||||||
host = etree.Element('host', name=name)
|
|
||||||
diff --git a/nova/virt/libvirt/driver.py b/nova/virt/libvirt/driver.py
|
|
||||||
index fbd033690a..74dc59ce87 100644
|
|
||||||
--- a/nova/virt/libvirt/driver.py
|
|
||||||
+++ b/nova/virt/libvirt/driver.py
|
|
||||||
@@ -180,6 +180,7 @@ libvirt_volume_drivers = [
|
|
||||||
'local=nova.virt.libvirt.volume.volume.LibvirtVolumeDriver',
|
|
||||||
'fake=nova.virt.libvirt.volume.volume.LibvirtFakeVolumeDriver',
|
|
||||||
'rbd=nova.virt.libvirt.volume.net.LibvirtNetVolumeDriver',
|
|
||||||
+ 'vitastor=nova.virt.libvirt.volume.vitastor.LibvirtVitastorVolumeDriver',
|
|
||||||
'nfs=nova.virt.libvirt.volume.nfs.LibvirtNFSVolumeDriver',
|
|
||||||
'smbfs=nova.virt.libvirt.volume.smbfs.LibvirtSMBFSVolumeDriver',
|
|
||||||
'fibre_channel='
|
|
||||||
@@ -287,10 +288,10 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# This prevents the risk of one test setting a capability
|
|
||||||
# which bleeds over into other tests.
|
|
||||||
|
|
||||||
- # LVM and RBD require raw images. If we are not configured to
|
|
||||||
+ # LVM, RBD, Vitastor require raw images. If we are not configured to
|
|
||||||
# force convert images into raw format, then we _require_ raw
|
|
||||||
# images only.
|
|
||||||
- raw_only = ('rbd', 'lvm')
|
|
||||||
+ raw_only = ('rbd', 'lvm', 'vitastor')
|
|
||||||
requires_raw_image = (CONF.libvirt.images_type in raw_only and
|
|
||||||
not CONF.force_raw_images)
|
|
||||||
requires_ploop_image = CONF.libvirt.virt_type == 'parallels'
|
|
||||||
@@ -703,12 +704,12 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# Some imagebackends are only able to import raw disk images,
|
|
||||||
# and will fail if given any other format. See the bug
|
|
||||||
# https://bugs.launchpad.net/nova/+bug/1816686 for more details.
|
|
||||||
- if CONF.libvirt.images_type in ('rbd',):
|
|
||||||
+ if CONF.libvirt.images_type in ('rbd', 'vitastor'):
|
|
||||||
if not CONF.force_raw_images:
|
|
||||||
msg = _("'[DEFAULT]/force_raw_images = False' is not "
|
|
||||||
- "allowed with '[libvirt]/images_type = rbd'. "
|
|
||||||
+ "allowed with '[libvirt]/images_type = rbd' or 'vitastor'. "
|
|
||||||
"Please check the two configs and if you really "
|
|
||||||
- "do want to use rbd as images_type, set "
|
|
||||||
+ "do want to use rbd or vitastor as images_type, set "
|
|
||||||
"force_raw_images to True.")
|
|
||||||
raise exception.InvalidConfiguration(msg)
|
|
||||||
|
|
||||||
@@ -2165,6 +2166,16 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
if connection_info['data'].get('auth_enabled'):
|
|
||||||
username = connection_info['data']['auth_username']
|
|
||||||
path = f"rbd:{volume_name}:id={username}"
|
|
||||||
+ elif connection_info['driver_volume_type'] == 'vitastor':
|
|
||||||
+ volume_name = connection_info['data']['name']
|
|
||||||
+ path = 'vitastor:image='+volume_name.replace(':', '\\:')
|
|
||||||
+ for k in [ 'config_path', 'etcd_address', 'etcd_prefix' ]:
|
|
||||||
+ if k in connection_info['data']:
|
|
||||||
+ kk = k
|
|
||||||
+ if kk == 'etcd_address':
|
|
||||||
+ # FIXME use etcd_address in qemu driver
|
|
||||||
+ kk = 'etcd_host'
|
|
||||||
+ path += ":"+kk.replace('_', '-')+"="+connection_info['data'][k].replace(':', '\\:')
|
|
||||||
else:
|
|
||||||
path = 'unknown'
|
|
||||||
raise exception.DiskNotFound(location='unknown')
|
|
||||||
@@ -2440,8 +2451,8 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
|
|
||||||
image_format = CONF.libvirt.snapshot_image_format or source_type
|
|
||||||
|
|
||||||
- # NOTE(bfilippov): save lvm and rbd as raw
|
|
||||||
- if image_format == 'lvm' or image_format == 'rbd':
|
|
||||||
+ # NOTE(bfilippov): save lvm and rbd and vitastor as raw
|
|
||||||
+ if image_format == 'lvm' or image_format == 'rbd' or image_format == 'vitastor':
|
|
||||||
image_format = 'raw'
|
|
||||||
|
|
||||||
metadata = self._create_snapshot_metadata(instance.image_meta,
|
|
||||||
@@ -2512,7 +2523,7 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
expected_state=task_states.IMAGE_UPLOADING)
|
|
||||||
|
|
||||||
# TODO(nic): possibly abstract this out to the root_disk
|
|
||||||
- if source_type == 'rbd' and live_snapshot:
|
|
||||||
+ if (source_type == 'rbd' or source_type == 'vitastor') and live_snapshot:
|
|
||||||
# Standard snapshot uses qemu-img convert from RBD which is
|
|
||||||
# not safe to run with live_snapshot.
|
|
||||||
live_snapshot = False
|
|
||||||
@@ -3715,7 +3726,7 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# cleanup rescue volume
|
|
||||||
lvm.remove_volumes([lvmdisk for lvmdisk in self._lvm_disks(instance)
|
|
||||||
if lvmdisk.endswith('.rescue')])
|
|
||||||
- if CONF.libvirt.images_type == 'rbd':
|
|
||||||
+ if CONF.libvirt.images_type == 'rbd' or CONF.libvirt.images_type == 'vitastor':
|
|
||||||
filter_fn = lambda disk: (disk.startswith(instance.uuid) and
|
|
||||||
disk.endswith('.rescue'))
|
|
||||||
rbd_utils.RBDDriver().cleanup_volumes(filter_fn)
|
|
||||||
@@ -3972,6 +3983,8 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# TODO(mikal): there is a bug here if images_type has
|
|
||||||
# changed since creation of the instance, but I am pretty
|
|
||||||
# sure that this bug already exists.
|
|
||||||
+ if CONF.libvirt.images_type == 'vitastor':
|
|
||||||
+ return 'vitastor'
|
|
||||||
return 'rbd' if CONF.libvirt.images_type == 'rbd' else 'raw'
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
@@ -4370,10 +4383,10 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
finally:
|
|
||||||
# NOTE(mikal): if the config drive was imported into RBD,
|
|
||||||
# then we no longer need the local copy
|
|
||||||
- if CONF.libvirt.images_type == 'rbd':
|
|
||||||
+ if CONF.libvirt.images_type == 'rbd' or CONF.libvirt.images_type == 'vitastor':
|
|
||||||
LOG.info('Deleting local config drive %(path)s '
|
|
||||||
- 'because it was imported into RBD.',
|
|
||||||
- {'path': config_disk_local_path},
|
|
||||||
+ 'because it was imported into %(type).',
|
|
||||||
+ {'path': config_disk_local_path, 'type': CONF.libvirt.images_type},
|
|
||||||
instance=instance)
|
|
||||||
os.unlink(config_disk_local_path)
|
|
||||||
|
|
||||||
diff --git a/nova/virt/libvirt/utils.py b/nova/virt/libvirt/utils.py
|
|
||||||
index c1dc34daf4..263965912f 100644
|
|
||||||
--- a/nova/virt/libvirt/utils.py
|
|
||||||
+++ b/nova/virt/libvirt/utils.py
|
|
||||||
@@ -399,6 +399,10 @@ def find_disk(guest: libvirt_guest.Guest) -> ty.Tuple[str, ty.Optional[str]]:
|
|
||||||
disk_path = disk.source_name
|
|
||||||
if disk_path:
|
|
||||||
disk_path = 'rbd:' + disk_path
|
|
||||||
+ elif not disk_path and disk.source_protocol == 'vitastor':
|
|
||||||
+ disk_path = disk.source_name
|
|
||||||
+ if disk_path:
|
|
||||||
+ disk_path = 'vitastor:' + disk_path
|
|
||||||
|
|
||||||
if not disk_path:
|
|
||||||
raise RuntimeError(_("Can't retrieve root device path "
|
|
||||||
@@ -417,6 +421,8 @@ def get_disk_type_from_path(path: str) -> ty.Optional[str]:
|
|
||||||
return 'lvm'
|
|
||||||
elif path.startswith('rbd:'):
|
|
||||||
return 'rbd'
|
|
||||||
+ elif path.startswith('vitastor:'):
|
|
||||||
+ return 'vitastor'
|
|
||||||
elif (os.path.isdir(path) and
|
|
||||||
os.path.exists(os.path.join(path, "DiskDescriptor.xml"))):
|
|
||||||
return 'ploop'
|
|
||||||
diff --git a/nova/virt/libvirt/volume/vitastor.py b/nova/virt/libvirt/volume/vitastor.py
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000..0256df62c1
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/nova/virt/libvirt/volume/vitastor.py
|
|
||||||
@@ -0,0 +1,75 @@
|
|
||||||
+# Copyright (c) 2021+, Vitaliy Filippov <vitalif@yourcmc.ru>
|
|
||||||
+#
|
|
||||||
+# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
||||||
+# not use this file except in compliance with the License. You may obtain
|
|
||||||
+# a copy of the License at
|
|
||||||
+#
|
|
||||||
+# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
+#
|
|
||||||
+# Unless required by applicable law or agreed to in writing, software
|
|
||||||
+# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
+# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
+# License for the specific language governing permissions and limitations
|
|
||||||
+# under the License.
|
|
||||||
+
|
|
||||||
+from os_brick import exception as os_brick_exception
|
|
||||||
+from os_brick import initiator
|
|
||||||
+from os_brick.initiator import connector
|
|
||||||
+from oslo_log import log as logging
|
|
||||||
+
|
|
||||||
+import nova.conf
|
|
||||||
+from nova import utils
|
|
||||||
+from nova.virt.libvirt.volume import volume as libvirt_volume
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+CONF = nova.conf.CONF
|
|
||||||
+LOG = logging.getLogger(__name__)
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+class LibvirtVitastorVolumeDriver(libvirt_volume.LibvirtBaseVolumeDriver):
|
|
||||||
+ """Driver to attach Vitastor volumes to libvirt."""
|
|
||||||
+ def __init__(self, host):
|
|
||||||
+ super(LibvirtVitastorVolumeDriver, self).__init__(host, is_block_dev=False)
|
|
||||||
+
|
|
||||||
+ def connect_volume(self, connection_info, instance):
|
|
||||||
+ pass
|
|
||||||
+
|
|
||||||
+ def disconnect_volume(self, connection_info, instance):
|
|
||||||
+ pass
|
|
||||||
+
|
|
||||||
+ def get_config(self, connection_info, disk_info):
|
|
||||||
+ """Returns xml for libvirt."""
|
|
||||||
+ conf = super(LibvirtVitastorVolumeDriver, self).get_config(connection_info, disk_info)
|
|
||||||
+ conf.source_type = 'network'
|
|
||||||
+ conf.source_protocol = 'vitastor'
|
|
||||||
+ conf.source_name = connection_info['data'].get('name')
|
|
||||||
+ conf.source_query = connection_info['data'].get('etcd_prefix') or None
|
|
||||||
+ conf.source_config = connection_info['data'].get('config_path') or None
|
|
||||||
+ conf.source_hosts = []
|
|
||||||
+ conf.source_ports = []
|
|
||||||
+ addresses = connection_info['data'].get('etcd_address', '')
|
|
||||||
+ if addresses:
|
|
||||||
+ if not isinstance(addresses, list):
|
|
||||||
+ addresses = addresses.split(',')
|
|
||||||
+ for addr in addresses:
|
|
||||||
+ if addr.startswith('https://'):
|
|
||||||
+ raise NotImplementedError('Vitastor block driver does not support SSL for etcd communication yet')
|
|
||||||
+ if addr.startswith('http://'):
|
|
||||||
+ addr = addr[7:]
|
|
||||||
+ addr = addr.rstrip('/')
|
|
||||||
+ if addr.endswith('/v3'):
|
|
||||||
+ addr = addr[0:-3]
|
|
||||||
+ p = addr.find('/')
|
|
||||||
+ if p > 0:
|
|
||||||
+ raise NotImplementedError('libvirt does not support custom URL paths for Vitastor etcd yet. Use /etc/vitastor/vitastor.conf')
|
|
||||||
+ p = addr.find(':')
|
|
||||||
+ port = '2379'
|
|
||||||
+ if p > 0:
|
|
||||||
+ port = addr[p+1:]
|
|
||||||
+ addr = addr[0:p]
|
|
||||||
+ conf.source_hosts.append(addr)
|
|
||||||
+ conf.source_ports.append(port)
|
|
||||||
+ return conf
|
|
||||||
+
|
|
||||||
+ def extend_volume(self, connection_info, instance, requested_size):
|
|
||||||
+ raise NotImplementedError
|
|
|
@ -1,288 +0,0 @@
|
||||||
diff --git a/nova/virt/image/model.py b/nova/virt/image/model.py
|
|
||||||
index 971f7e9c07..ec3fca72cb 100644
|
|
||||||
--- a/nova/virt/image/model.py
|
|
||||||
+++ b/nova/virt/image/model.py
|
|
||||||
@@ -129,3 +129,22 @@ class RBDImage(Image):
|
|
||||||
self.user = user
|
|
||||||
self.password = password
|
|
||||||
self.servers = servers
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+class VitastorImage(Image):
|
|
||||||
+ """Class for images in a remote Vitastor cluster"""
|
|
||||||
+
|
|
||||||
+ def __init__(self, name, etcd_address = None, etcd_prefix = None, config_path = None):
|
|
||||||
+ """Create a new Vitastor image object
|
|
||||||
+
|
|
||||||
+ :param name: name of the image
|
|
||||||
+ :param etcd_address: etcd URL(s) (optional)
|
|
||||||
+ :param etcd_prefix: etcd prefix (optional)
|
|
||||||
+ :param config_path: path to the configuration (optional)
|
|
||||||
+ """
|
|
||||||
+ super(VitastorImage, self).__init__(FORMAT_RAW)
|
|
||||||
+
|
|
||||||
+ self.name = name
|
|
||||||
+ self.etcd_address = etcd_address
|
|
||||||
+ self.etcd_prefix = etcd_prefix
|
|
||||||
+ self.config_path = config_path
|
|
||||||
diff --git a/nova/virt/images.py b/nova/virt/images.py
|
|
||||||
index 5358f3766a..ebe3d6effb 100644
|
|
||||||
--- a/nova/virt/images.py
|
|
||||||
+++ b/nova/virt/images.py
|
|
||||||
@@ -41,7 +41,7 @@ IMAGE_API = glance.API()
|
|
||||||
|
|
||||||
def qemu_img_info(path, format=None):
|
|
||||||
"""Return an object containing the parsed output from qemu-img info."""
|
|
||||||
- if not os.path.exists(path) and not path.startswith('rbd:'):
|
|
||||||
+ if not os.path.exists(path) and not path.startswith('rbd:') and not path.startswith('vitastor:'):
|
|
||||||
raise exception.DiskNotFound(location=path)
|
|
||||||
|
|
||||||
info = nova.privsep.qemu.unprivileged_qemu_img_info(path, format=format)
|
|
||||||
@@ -50,7 +50,7 @@ def qemu_img_info(path, format=None):
|
|
||||||
|
|
||||||
def privileged_qemu_img_info(path, format=None, output_format='json'):
|
|
||||||
"""Return an object containing the parsed output from qemu-img info."""
|
|
||||||
- if not os.path.exists(path) and not path.startswith('rbd:'):
|
|
||||||
+ if not os.path.exists(path) and not path.startswith('rbd:') and not path.startswith('vitastor:'):
|
|
||||||
raise exception.DiskNotFound(location=path)
|
|
||||||
|
|
||||||
info = nova.privsep.qemu.privileged_qemu_img_info(path, format=format)
|
|
||||||
diff --git a/nova/virt/libvirt/config.py b/nova/virt/libvirt/config.py
|
|
||||||
index f9475776b3..a2e18aab67 100644
|
|
||||||
--- a/nova/virt/libvirt/config.py
|
|
||||||
+++ b/nova/virt/libvirt/config.py
|
|
||||||
@@ -1060,6 +1060,8 @@ class LibvirtConfigGuestDisk(LibvirtConfigGuestDevice):
|
|
||||||
self.driver_iommu = False
|
|
||||||
self.source_path = None
|
|
||||||
self.source_protocol = None
|
|
||||||
+ self.source_query = None
|
|
||||||
+ self.source_config = None
|
|
||||||
self.source_name = None
|
|
||||||
self.source_hosts = []
|
|
||||||
self.source_ports = []
|
|
||||||
@@ -1189,6 +1191,10 @@ class LibvirtConfigGuestDisk(LibvirtConfigGuestDevice):
|
|
||||||
source = etree.Element("source", protocol=self.source_protocol)
|
|
||||||
if self.source_name is not None:
|
|
||||||
source.set('name', self.source_name)
|
|
||||||
+ if self.source_query is not None:
|
|
||||||
+ source.set('query', self.source_query)
|
|
||||||
+ if self.source_config is not None:
|
|
||||||
+ source.append(etree.Element('config', file=self.source_config))
|
|
||||||
hosts_info = zip(self.source_hosts, self.source_ports)
|
|
||||||
for name, port in hosts_info:
|
|
||||||
host = etree.Element('host', name=name)
|
|
||||||
diff --git a/nova/virt/libvirt/driver.py b/nova/virt/libvirt/driver.py
|
|
||||||
index 391231c527..f38faa1608 100644
|
|
||||||
--- a/nova/virt/libvirt/driver.py
|
|
||||||
+++ b/nova/virt/libvirt/driver.py
|
|
||||||
@@ -179,6 +179,7 @@ VOLUME_DRIVERS = {
|
|
||||||
'local': 'nova.virt.libvirt.volume.volume.LibvirtVolumeDriver',
|
|
||||||
'fake': 'nova.virt.libvirt.volume.volume.LibvirtFakeVolumeDriver',
|
|
||||||
'rbd': 'nova.virt.libvirt.volume.net.LibvirtNetVolumeDriver',
|
|
||||||
+ 'vitastor': 'nova.virt.libvirt.volume.vitastor.LibvirtVitastorVolumeDriver',
|
|
||||||
'nfs': 'nova.virt.libvirt.volume.nfs.LibvirtNFSVolumeDriver',
|
|
||||||
'smbfs': 'nova.virt.libvirt.volume.smbfs.LibvirtSMBFSVolumeDriver',
|
|
||||||
'fibre_channel': 'nova.virt.libvirt.volume.fibrechannel.LibvirtFibreChannelVolumeDriver', # noqa:E501
|
|
||||||
@@ -385,10 +386,10 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# This prevents the risk of one test setting a capability
|
|
||||||
# which bleeds over into other tests.
|
|
||||||
|
|
||||||
- # LVM and RBD require raw images. If we are not configured to
|
|
||||||
+ # LVM, RBD, Vitastor require raw images. If we are not configured to
|
|
||||||
# force convert images into raw format, then we _require_ raw
|
|
||||||
# images only.
|
|
||||||
- raw_only = ('rbd', 'lvm')
|
|
||||||
+ raw_only = ('rbd', 'lvm', 'vitastor')
|
|
||||||
requires_raw_image = (CONF.libvirt.images_type in raw_only and
|
|
||||||
not CONF.force_raw_images)
|
|
||||||
requires_ploop_image = CONF.libvirt.virt_type == 'parallels'
|
|
||||||
@@ -775,12 +776,12 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# Some imagebackends are only able to import raw disk images,
|
|
||||||
# and will fail if given any other format. See the bug
|
|
||||||
# https://bugs.launchpad.net/nova/+bug/1816686 for more details.
|
|
||||||
- if CONF.libvirt.images_type in ('rbd',):
|
|
||||||
+ if CONF.libvirt.images_type in ('rbd', 'vitastor'):
|
|
||||||
if not CONF.force_raw_images:
|
|
||||||
msg = _("'[DEFAULT]/force_raw_images = False' is not "
|
|
||||||
- "allowed with '[libvirt]/images_type = rbd'. "
|
|
||||||
+ "allowed with '[libvirt]/images_type = rbd' or 'vitastor'. "
|
|
||||||
"Please check the two configs and if you really "
|
|
||||||
- "do want to use rbd as images_type, set "
|
|
||||||
+ "do want to use rbd or vitastor as images_type, set "
|
|
||||||
"force_raw_images to True.")
|
|
||||||
raise exception.InvalidConfiguration(msg)
|
|
||||||
|
|
||||||
@@ -2603,6 +2604,16 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
if connection_info['data'].get('auth_enabled'):
|
|
||||||
username = connection_info['data']['auth_username']
|
|
||||||
path = f"rbd:{volume_name}:id={username}"
|
|
||||||
+ elif connection_info['driver_volume_type'] == 'vitastor':
|
|
||||||
+ volume_name = connection_info['data']['name']
|
|
||||||
+ path = 'vitastor:image='+volume_name.replace(':', '\\:')
|
|
||||||
+ for k in [ 'config_path', 'etcd_address', 'etcd_prefix' ]:
|
|
||||||
+ if k in connection_info['data']:
|
|
||||||
+ kk = k
|
|
||||||
+ if kk == 'etcd_address':
|
|
||||||
+ # FIXME use etcd_address in qemu driver
|
|
||||||
+ kk = 'etcd_host'
|
|
||||||
+ path += ":"+kk.replace('_', '-')+"="+connection_info['data'][k].replace(':', '\\:')
|
|
||||||
else:
|
|
||||||
path = 'unknown'
|
|
||||||
raise exception.DiskNotFound(location='unknown')
|
|
||||||
@@ -2827,8 +2838,8 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
|
|
||||||
image_format = CONF.libvirt.snapshot_image_format or source_type
|
|
||||||
|
|
||||||
- # NOTE(bfilippov): save lvm and rbd as raw
|
|
||||||
- if image_format == 'lvm' or image_format == 'rbd':
|
|
||||||
+ # NOTE(bfilippov): save lvm and rbd and vitastor as raw
|
|
||||||
+ if image_format == 'lvm' or image_format == 'rbd' or image_format == 'vitastor':
|
|
||||||
image_format = 'raw'
|
|
||||||
|
|
||||||
metadata = self._create_snapshot_metadata(instance.image_meta,
|
|
||||||
@@ -2899,7 +2910,7 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
expected_state=task_states.IMAGE_UPLOADING)
|
|
||||||
|
|
||||||
# TODO(nic): possibly abstract this out to the root_disk
|
|
||||||
- if source_type == 'rbd' and live_snapshot:
|
|
||||||
+ if (source_type == 'rbd' or source_type == 'vitastor') and live_snapshot:
|
|
||||||
# Standard snapshot uses qemu-img convert from RBD which is
|
|
||||||
# not safe to run with live_snapshot.
|
|
||||||
live_snapshot = False
|
|
||||||
@@ -4099,7 +4110,7 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# cleanup rescue volume
|
|
||||||
lvm.remove_volumes([lvmdisk for lvmdisk in self._lvm_disks(instance)
|
|
||||||
if lvmdisk.endswith('.rescue')])
|
|
||||||
- if CONF.libvirt.images_type == 'rbd':
|
|
||||||
+ if CONF.libvirt.images_type == 'rbd' or CONF.libvirt.images_type == 'vitastor':
|
|
||||||
filter_fn = lambda disk: (disk.startswith(instance.uuid) and
|
|
||||||
disk.endswith('.rescue'))
|
|
||||||
rbd_utils.RBDDriver().cleanup_volumes(filter_fn)
|
|
||||||
@@ -4356,6 +4367,8 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
# TODO(mikal): there is a bug here if images_type has
|
|
||||||
# changed since creation of the instance, but I am pretty
|
|
||||||
# sure that this bug already exists.
|
|
||||||
+ if CONF.libvirt.images_type == 'vitastor':
|
|
||||||
+ return 'vitastor'
|
|
||||||
return 'rbd' if CONF.libvirt.images_type == 'rbd' else 'raw'
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
@@ -4764,10 +4777,10 @@ class LibvirtDriver(driver.ComputeDriver):
|
|
||||||
finally:
|
|
||||||
# NOTE(mikal): if the config drive was imported into RBD,
|
|
||||||
# then we no longer need the local copy
|
|
||||||
- if CONF.libvirt.images_type == 'rbd':
|
|
||||||
+ if CONF.libvirt.images_type == 'rbd' or CONF.libvirt.images_type == 'vitastor':
|
|
||||||
LOG.info('Deleting local config drive %(path)s '
|
|
||||||
- 'because it was imported into RBD.',
|
|
||||||
- {'path': config_disk_local_path},
|
|
||||||
+ 'because it was imported into %(type).',
|
|
||||||
+ {'path': config_disk_local_path, 'type': CONF.libvirt.images_type},
|
|
||||||
instance=instance)
|
|
||||||
os.unlink(config_disk_local_path)
|
|
||||||
|
|
||||||
diff --git a/nova/virt/libvirt/utils.py b/nova/virt/libvirt/utils.py
|
|
||||||
index da2a6e8b8a..52c02e72f1 100644
|
|
||||||
--- a/nova/virt/libvirt/utils.py
|
|
||||||
+++ b/nova/virt/libvirt/utils.py
|
|
||||||
@@ -340,6 +340,10 @@ def find_disk(guest: libvirt_guest.Guest) -> ty.Tuple[str, ty.Optional[str]]:
|
|
||||||
disk_path = disk.source_name
|
|
||||||
if disk_path:
|
|
||||||
disk_path = 'rbd:' + disk_path
|
|
||||||
+ elif not disk_path and disk.source_protocol == 'vitastor':
|
|
||||||
+ disk_path = disk.source_name
|
|
||||||
+ if disk_path:
|
|
||||||
+ disk_path = 'vitastor:' + disk_path
|
|
||||||
|
|
||||||
if not disk_path:
|
|
||||||
raise RuntimeError(_("Can't retrieve root device path "
|
|
||||||
@@ -354,6 +358,8 @@ def get_disk_type_from_path(path: str) -> ty.Optional[str]:
|
|
||||||
return 'lvm'
|
|
||||||
elif path.startswith('rbd:'):
|
|
||||||
return 'rbd'
|
|
||||||
+ elif path.startswith('vitastor:'):
|
|
||||||
+ return 'vitastor'
|
|
||||||
elif (os.path.isdir(path) and
|
|
||||||
os.path.exists(os.path.join(path, "DiskDescriptor.xml"))):
|
|
||||||
return 'ploop'
|
|
||||||
diff --git a/nova/virt/libvirt/volume/vitastor.py b/nova/virt/libvirt/volume/vitastor.py
|
|
||||||
new file mode 100644
|
|
||||||
index 0000000000..0256df62c1
|
|
||||||
--- /dev/null
|
|
||||||
+++ b/nova/virt/libvirt/volume/vitastor.py
|
|
||||||
@@ -0,0 +1,75 @@
|
|
||||||
+# Copyright (c) 2021+, Vitaliy Filippov <vitalif@yourcmc.ru>
|
|
||||||
+#
|
|
||||||
+# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
||||||
+# not use this file except in compliance with the License. You may obtain
|
|
||||||
+# a copy of the License at
|
|
||||||
+#
|
|
||||||
+# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
+#
|
|
||||||
+# Unless required by applicable law or agreed to in writing, software
|
|
||||||
+# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
+# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
+# License for the specific language governing permissions and limitations
|
|
||||||
+# under the License.
|
|
||||||
+
|
|
||||||
+from os_brick import exception as os_brick_exception
|
|
||||||
+from os_brick import initiator
|
|
||||||
+from os_brick.initiator import connector
|
|
||||||
+from oslo_log import log as logging
|
|
||||||
+
|
|
||||||
+import nova.conf
|
|
||||||
+from nova import utils
|
|
||||||
+from nova.virt.libvirt.volume import volume as libvirt_volume
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+CONF = nova.conf.CONF
|
|
||||||
+LOG = logging.getLogger(__name__)
|
|
||||||
+
|
|
||||||
+
|
|
||||||
+class LibvirtVitastorVolumeDriver(libvirt_volume.LibvirtBaseVolumeDriver):
|
|
||||||
+ """Driver to attach Vitastor volumes to libvirt."""
|
|
||||||
+ def __init__(self, host):
|
|
||||||
+ super(LibvirtVitastorVolumeDriver, self).__init__(host, is_block_dev=False)
|
|
||||||
+
|
|
||||||
+ def connect_volume(self, connection_info, instance):
|
|
||||||
+ pass
|
|
||||||
+
|
|
||||||
+ def disconnect_volume(self, connection_info, instance):
|
|
||||||
+ pass
|
|
||||||
+
|
|
||||||
+ def get_config(self, connection_info, disk_info):
|
|
||||||
+ """Returns xml for libvirt."""
|
|
||||||
+ conf = super(LibvirtVitastorVolumeDriver, self).get_config(connection_info, disk_info)
|
|
||||||
+ conf.source_type = 'network'
|
|
||||||
+ conf.source_protocol = 'vitastor'
|
|
||||||
+ conf.source_name = connection_info['data'].get('name')
|
|
||||||
+ conf.source_query = connection_info['data'].get('etcd_prefix') or None
|
|
||||||
+ conf.source_config = connection_info['data'].get('config_path') or None
|
|
||||||
+ conf.source_hosts = []
|
|
||||||
+ conf.source_ports = []
|
|
||||||
+ addresses = connection_info['data'].get('etcd_address', '')
|
|
||||||
+ if addresses:
|
|
||||||
+ if not isinstance(addresses, list):
|
|
||||||
+ addresses = addresses.split(',')
|
|
||||||
+ for addr in addresses:
|
|
||||||
+ if addr.startswith('https://'):
|
|
||||||
+ raise NotImplementedError('Vitastor block driver does not support SSL for etcd communication yet')
|
|
||||||
+ if addr.startswith('http://'):
|
|
||||||
+ addr = addr[7:]
|
|
||||||
+ addr = addr.rstrip('/')
|
|
||||||
+ if addr.endswith('/v3'):
|
|
||||||
+ addr = addr[0:-3]
|
|
||||||
+ p = addr.find('/')
|
|
||||||
+ if p > 0:
|
|
||||||
+ raise NotImplementedError('libvirt does not support custom URL paths for Vitastor etcd yet. Use /etc/vitastor/vitastor.conf')
|
|
||||||
+ p = addr.find(':')
|
|
||||||
+ port = '2379'
|
|
||||||
+ if p > 0:
|
|
||||||
+ port = addr[p+1:]
|
|
||||||
+ addr = addr[0:p]
|
|
||||||
+ conf.source_hosts.append(addr)
|
|
||||||
+ conf.source_ports.append(port)
|
|
||||||
+ return conf
|
|
||||||
+
|
|
||||||
+ def extend_volume(self, connection_info, instance, requested_size):
|
|
||||||
+ raise NotImplementedError
|
|
|
@ -1,175 +0,0 @@
|
||||||
Index: pve-qemu-kvm-5.1.0/qapi/block-core.json
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-5.1.0.orig/qapi/block-core.json
|
|
||||||
+++ pve-qemu-kvm-5.1.0/qapi/block-core.json
|
|
||||||
@@ -3041,7 +3041,7 @@
|
|
||||||
'luks', 'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels',
|
|
||||||
'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
- 'sheepdog', 'pbs',
|
|
||||||
+ 'sheepdog', 'pbs', 'vitastor',
|
|
||||||
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
@@ -3889,6 +3889,28 @@
|
|
||||||
'*tag': 'str' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
# An enumeration of replication modes.
|
|
||||||
@@ -4234,6 +4256,7 @@
|
|
||||||
'replication': { 'type': 'BlockdevOptionsReplication',
|
|
||||||
'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
'sheepdog': 'BlockdevOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevOptionsSsh',
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
@@ -4623,6 +4646,17 @@
|
|
||||||
'*cluster-size' : 'size' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
# Subformat options for VMDK images
|
|
||||||
@@ -4884,6 +4918,7 @@
|
|
||||||
'qed': 'BlockdevCreateOptionsQed',
|
|
||||||
'rbd': 'BlockdevCreateOptionsRbd',
|
|
||||||
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
||||||
Index: pve-qemu-kvm-5.1.0/configure
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-5.1.0.orig/configure
|
|
||||||
+++ pve-qemu-kvm-5.1.0/configure
|
|
||||||
@@ -446,6 +446,7 @@ trace_backends="log"
|
|
||||||
trace_file="trace"
|
|
||||||
spice=""
|
|
||||||
rbd=""
|
|
||||||
+vitastor=""
|
|
||||||
smartcard=""
|
|
||||||
libusb=""
|
|
||||||
usb_redir=""
|
|
||||||
@@ -1383,6 +1384,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="yes"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="no"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="yes"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1901,6 +1906,7 @@ disabled with --disable-FEATURE, default
|
|
||||||
vhost-vdpa vhost-vdpa kernel backend support
|
|
||||||
spice spice
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -4234,6 +4240,27 @@ EOF
|
|
||||||
fi
|
|
||||||
|
|
||||||
##########################################
|
|
||||||
+# vitastor probe
|
|
||||||
+if test "$vitastor" != "no" ; then
|
|
||||||
+ cat > $TMPC <<EOF
|
|
||||||
+#include <vitastor_c.h>
|
|
||||||
+int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+EOF
|
|
||||||
+ vitastor_libs="-lvitastor_client"
|
|
||||||
+ if compile_prog "" "$vitastor_libs" ; then
|
|
||||||
+ vitastor=yes
|
|
||||||
+ else
|
|
||||||
+ if test "$vitastor" = "yes" ; then
|
|
||||||
+ feature_not_found "vitastor block device" "Install vitastor-client-dev"
|
|
||||||
+ fi
|
|
||||||
+ vitastor=no
|
|
||||||
+ fi
|
|
||||||
+fi
|
|
||||||
+
|
|
||||||
+##########################################
|
|
||||||
# libssh probe
|
|
||||||
if test "$libssh" != "no" ; then
|
|
||||||
if $pkg_config --exists libssh; then
|
|
||||||
@@ -6969,6 +6996,7 @@ echo "Trace output file $trace_file-<pid
|
|
||||||
fi
|
|
||||||
echo "spice support $spice $(echo_version $spice $spice_protocol_version/$spice_server_version)"
|
|
||||||
echo "rbd support $rbd"
|
|
||||||
+echo "vitastor support $vitastor"
|
|
||||||
echo "xfsctl support $xfs"
|
|
||||||
echo "smartcard support $smartcard"
|
|
||||||
echo "libusb $libusb"
|
|
||||||
@@ -7644,6 +7672,10 @@ if test "$rbd" = "yes" ; then
|
|
||||||
echo "RBD_CFLAGS=$rbd_cflags" >> $config_host_mak
|
|
||||||
echo "RBD_LIBS=$rbd_libs" >> $config_host_mak
|
|
||||||
fi
|
|
||||||
+if test "$vitastor" = "yes" ; then
|
|
||||||
+ echo "CONFIG_VITASTOR=y" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_LIBS=$vitastor_libs" >> $config_host_mak
|
|
||||||
+fi
|
|
||||||
|
|
||||||
echo "CONFIG_COROUTINE_BACKEND=$coroutine" >> $config_host_mak
|
|
||||||
if test "$coroutine_pool" = "yes" ; then
|
|
||||||
Index: pve-qemu-kvm-5.1.0/block/Makefile.objs
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-5.1.0.orig/block/Makefile.objs
|
|
||||||
+++ pve-qemu-kvm-5.1.0/block/Makefile.objs
|
|
||||||
@@ -32,6 +32,7 @@ block-obj-$(if $(CONFIG_LIBISCSI),y,n) +
|
|
||||||
block-obj-$(CONFIG_LIBNFS) += nfs.o
|
|
||||||
block-obj-$(CONFIG_CURL) += curl.o
|
|
||||||
block-obj-$(CONFIG_RBD) += rbd.o
|
|
||||||
+block-obj-$(CONFIG_VITASTOR) += vitastor.o
|
|
||||||
block-obj-$(CONFIG_GLUSTERFS) += gluster.o
|
|
||||||
block-obj-$(CONFIG_LIBSSH) += ssh.o
|
|
||||||
block-obj-y += backup-dump.o
|
|
||||||
@@ -61,6 +62,8 @@ curl.o-cflags := $(CURL_CFLAGS)
|
|
||||||
curl.o-libs := $(CURL_LIBS)
|
|
||||||
rbd.o-cflags := $(RBD_CFLAGS)
|
|
||||||
rbd.o-libs := $(RBD_LIBS)
|
|
||||||
+vitastor.o-cflags := $(VITASTOR_CFLAGS)
|
|
||||||
+vitastor.o-libs := $(VITASTOR_LIBS)
|
|
||||||
gluster.o-cflags := $(GLUSTERFS_CFLAGS)
|
|
||||||
gluster.o-libs := $(GLUSTERFS_LIBS)
|
|
||||||
ssh.o-cflags := $(LIBSSH_CFLAGS)
|
|
|
@ -1,181 +0,0 @@
|
||||||
Index: pve-qemu-kvm-5.2.0/qapi/block-core.json
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-5.2.0.orig/qapi/block-core.json
|
|
||||||
+++ pve-qemu-kvm-5.2.0/qapi/block-core.json
|
|
||||||
@@ -3076,7 +3076,7 @@
|
|
||||||
'luks', 'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels',
|
|
||||||
'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
- 'sheepdog', 'pbs',
|
|
||||||
+ 'sheepdog', 'pbs', 'vitastor',
|
|
||||||
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
@@ -3924,6 +3924,28 @@
|
|
||||||
'*tag': 'str' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
# An enumeration of replication modes.
|
|
||||||
@@ -4272,6 +4294,7 @@
|
|
||||||
'replication': { 'type': 'BlockdevOptionsReplication',
|
|
||||||
'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
'sheepdog': 'BlockdevOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevOptionsSsh',
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
@@ -4662,6 +4685,17 @@
|
|
||||||
'*cluster-size' : 'size' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
# Subformat options for VMDK images
|
|
||||||
@@ -4923,6 +4957,7 @@
|
|
||||||
'qed': 'BlockdevCreateOptionsQed',
|
|
||||||
'rbd': 'BlockdevCreateOptionsRbd',
|
|
||||||
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
||||||
Index: pve-qemu-kvm-5.2.0/block/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-5.2.0.orig/block/meson.build
|
|
||||||
+++ pve-qemu-kvm-5.2.0/block/meson.build
|
|
||||||
@@ -89,6 +89,7 @@ foreach m : [
|
|
||||||
['CONFIG_LIBNFS', 'nfs', libnfs, 'nfs.c'],
|
|
||||||
['CONFIG_LIBSSH', 'ssh', libssh, 'ssh.c'],
|
|
||||||
['CONFIG_RBD', 'rbd', rbd, 'rbd.c'],
|
|
||||||
+ ['CONFIG_VITASTOR', 'vitastor', vitastor, 'vitastor.c'],
|
|
||||||
]
|
|
||||||
if config_host.has_key(m[0])
|
|
||||||
if enable_modules
|
|
||||||
Index: pve-qemu-kvm-5.2.0/configure
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-5.2.0.orig/configure
|
|
||||||
+++ pve-qemu-kvm-5.2.0/configure
|
|
||||||
@@ -372,6 +372,7 @@ trace_backends="log"
|
|
||||||
trace_file="trace"
|
|
||||||
spice=""
|
|
||||||
rbd=""
|
|
||||||
+vitastor=""
|
|
||||||
smartcard=""
|
|
||||||
u2f="auto"
|
|
||||||
libusb=""
|
|
||||||
@@ -1264,6 +1265,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="yes"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="no"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="yes"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1807,6 +1812,7 @@ disabled with --disable-FEATURE, default
|
|
||||||
vhost-vdpa vhost-vdpa kernel backend support
|
|
||||||
spice spice
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -3700,6 +3706,27 @@ EOF
|
|
||||||
fi
|
|
||||||
|
|
||||||
##########################################
|
|
||||||
+# vitastor probe
|
|
||||||
+if test "$vitastor" != "no" ; then
|
|
||||||
+ cat > $TMPC <<EOF
|
|
||||||
+#include <vitastor_c.h>
|
|
||||||
+int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+EOF
|
|
||||||
+ vitastor_libs="-lvitastor_client"
|
|
||||||
+ if compile_prog "" "$vitastor_libs" ; then
|
|
||||||
+ vitastor=yes
|
|
||||||
+ else
|
|
||||||
+ if test "$vitastor" = "yes" ; then
|
|
||||||
+ feature_not_found "vitastor block device" "Install vitastor-client-dev"
|
|
||||||
+ fi
|
|
||||||
+ vitastor=no
|
|
||||||
+ fi
|
|
||||||
+fi
|
|
||||||
+
|
|
||||||
+##########################################
|
|
||||||
# libssh probe
|
|
||||||
if test "$libssh" != "no" ; then
|
|
||||||
if $pkg_config --exists libssh; then
|
|
||||||
@@ -6437,6 +6464,10 @@ if test "$rbd" = "yes" ; then
|
|
||||||
echo "CONFIG_RBD=y" >> $config_host_mak
|
|
||||||
echo "RBD_LIBS=$rbd_libs" >> $config_host_mak
|
|
||||||
fi
|
|
||||||
+if test "$vitastor" = "yes" ; then
|
|
||||||
+ echo "CONFIG_VITASTOR=y" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_LIBS=$vitastor_libs" >> $config_host_mak
|
|
||||||
+fi
|
|
||||||
|
|
||||||
echo "CONFIG_COROUTINE_BACKEND=$coroutine" >> $config_host_mak
|
|
||||||
if test "$coroutine_pool" = "yes" ; then
|
|
||||||
Index: pve-qemu-kvm-5.2.0/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-5.2.0.orig/meson.build
|
|
||||||
+++ pve-qemu-kvm-5.2.0/meson.build
|
|
||||||
@@ -596,6 +596,10 @@ rbd = not_found
|
|
||||||
if 'CONFIG_RBD' in config_host
|
|
||||||
rbd = declare_dependency(link_args: config_host['RBD_LIBS'].split())
|
|
||||||
endif
|
|
||||||
+vitastor = not_found
|
|
||||||
+if 'CONFIG_VITASTOR' in config_host
|
|
||||||
+ vitastor = declare_dependency(link_args: config_host['VITASTOR_LIBS'].split())
|
|
||||||
+endif
|
|
||||||
glusterfs = not_found
|
|
||||||
if 'CONFIG_GLUSTERFS' in config_host
|
|
||||||
glusterfs = declare_dependency(compile_args: config_host['GLUSTERFS_CFLAGS'].split(),
|
|
||||||
@@ -2151,6 +2155,7 @@ endif
|
|
||||||
# TODO: add back protocol and server version
|
|
||||||
summary_info += {'spice support': config_host.has_key('CONFIG_SPICE')}
|
|
||||||
summary_info += {'rbd support': config_host.has_key('CONFIG_RBD')}
|
|
||||||
+summary_info += {'vitastor support': config_host.has_key('CONFIG_VITASTOR')}
|
|
||||||
summary_info += {'xfsctl support': config_host.has_key('CONFIG_XFS')}
|
|
||||||
summary_info += {'smartcard support': config_host.has_key('CONFIG_SMARTCARD')}
|
|
||||||
summary_info += {'U2F support': u2f.found()}
|
|
|
@ -1,188 +0,0 @@
|
||||||
Index: pve-qemu-kvm-6.1.0/qapi/block-core.json
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-6.1.0.orig/qapi/block-core.json
|
|
||||||
+++ pve-qemu-kvm-6.1.0/qapi/block-core.json
|
|
||||||
@@ -3084,7 +3084,7 @@
|
|
||||||
'preallocate', 'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
'pbs',
|
|
||||||
- 'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
+ 'ssh', 'throttle', 'vdi', 'vhdx', 'vitastor', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
# @BlockdevOptionsFile:
|
|
||||||
@@ -4020,6 +4020,28 @@
|
|
||||||
'*server': ['InetSocketAddressBase'] } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
# An enumeration of replication modes.
|
|
||||||
@@ -4392,6 +4414,7 @@
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
'vhdx': 'BlockdevOptionsGenericFormat',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'vmdk': 'BlockdevOptionsGenericCOWFormat',
|
|
||||||
'vpc': 'BlockdevOptionsGenericFormat',
|
|
||||||
'vvfat': 'BlockdevOptionsVVFAT'
|
|
||||||
@@ -4782,6 +4805,17 @@
|
|
||||||
'*encrypt' : 'RbdEncryptionCreateOptions' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
# Subformat options for VMDK images
|
|
||||||
@@ -4977,6 +5011,7 @@
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'vmdk': 'BlockdevCreateOptionsVmdk',
|
|
||||||
'vpc': 'BlockdevCreateOptionsVpc'
|
|
||||||
} }
|
|
||||||
Index: pve-qemu-kvm-6.1.0/block/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-6.1.0.orig/block/meson.build
|
|
||||||
+++ pve-qemu-kvm-6.1.0/block/meson.build
|
|
||||||
@@ -91,6 +91,7 @@ foreach m : [
|
|
||||||
[libnfs, 'nfs', files('nfs.c')],
|
|
||||||
[libssh, 'ssh', files('ssh.c')],
|
|
||||||
[rbd, 'rbd', files('rbd.c')],
|
|
||||||
+ [vitastor, 'vitastor', files('vitastor.c')],
|
|
||||||
]
|
|
||||||
if m[0].found()
|
|
||||||
module_ss = ss.source_set()
|
|
||||||
Index: pve-qemu-kvm-6.1.0/configure
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-6.1.0.orig/configure
|
|
||||||
+++ pve-qemu-kvm-6.1.0/configure
|
|
||||||
@@ -375,6 +375,7 @@ trace_file="trace"
|
|
||||||
spice="$default_feature"
|
|
||||||
spice_protocol="auto"
|
|
||||||
rbd="auto"
|
|
||||||
+vitastor="auto"
|
|
||||||
smartcard="auto"
|
|
||||||
u2f="auto"
|
|
||||||
libusb="auto"
|
|
||||||
@@ -1293,6 +1294,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="enabled"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="disabled"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="enabled"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1921,6 +1926,7 @@ disabled with --disable-FEATURE, default
|
|
||||||
spice spice
|
|
||||||
spice-protocol spice-protocol
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -5211,7 +5217,7 @@ if test "$skip_meson" = no; then
|
|
||||||
-Dcapstone=$capstone -Dslirp=$slirp -Dfdt=$fdt -Dbrlapi=$brlapi \
|
|
||||||
-Dcurl=$curl -Dglusterfs=$glusterfs -Dbzip2=$bzip2 -Dlibiscsi=$libiscsi \
|
|
||||||
-Dlibnfs=$libnfs -Diconv=$iconv -Dcurses=$curses -Dlibudev=$libudev\
|
|
||||||
- -Drbd=$rbd -Dlzo=$lzo -Dsnappy=$snappy -Dlzfse=$lzfse -Dlibxml2=$libxml2 \
|
|
||||||
+ -Drbd=$rbd -Dvitastor=$vitastor -Dlzo=$lzo -Dsnappy=$snappy -Dlzfse=$lzfse -Dlibxml2=$libxml2 \
|
|
||||||
-Dlibdaxctl=$libdaxctl -Dlibpmem=$libpmem -Dlinux_io_uring=$linux_io_uring \
|
|
||||||
-Dgnutls=$gnutls -Dnettle=$nettle -Dgcrypt=$gcrypt -Dauth_pam=$auth_pam \
|
|
||||||
-Dzstd=$zstd -Dseccomp=$seccomp -Dvirtfs=$virtfs -Dcap_ng=$cap_ng \
|
|
||||||
Index: pve-qemu-kvm-6.1.0/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-6.1.0.orig/meson.build
|
|
||||||
+++ pve-qemu-kvm-6.1.0/meson.build
|
|
||||||
@@ -729,6 +729,26 @@ if not get_option('rbd').auto() or have_
|
|
||||||
endif
|
|
||||||
endif
|
|
||||||
|
|
||||||
+vitastor = not_found
|
|
||||||
+if not get_option('vitastor').auto() or have_block
|
|
||||||
+ libvitastor_client = cc.find_library('vitastor_client', has_headers: ['vitastor_c.h'],
|
|
||||||
+ required: get_option('vitastor'), kwargs: static_kwargs)
|
|
||||||
+ if libvitastor_client.found()
|
|
||||||
+ if cc.links('''
|
|
||||||
+ #include <vitastor_c.h>
|
|
||||||
+ int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+ }''', dependencies: libvitastor_client)
|
|
||||||
+ vitastor = declare_dependency(dependencies: libvitastor_client)
|
|
||||||
+ elif get_option('vitastor').enabled()
|
|
||||||
+ error('could not link libvitastor_client')
|
|
||||||
+ else
|
|
||||||
+ warning('could not link libvitastor_client, disabling')
|
|
||||||
+ endif
|
|
||||||
+ endif
|
|
||||||
+endif
|
|
||||||
+
|
|
||||||
glusterfs = not_found
|
|
||||||
glusterfs_ftruncate_has_stat = false
|
|
||||||
glusterfs_iocb_has_stat = false
|
|
||||||
@@ -1268,6 +1288,7 @@ config_host_data.set('CONFIG_LIBNFS', li
|
|
||||||
config_host_data.set('CONFIG_LINUX_IO_URING', linux_io_uring.found())
|
|
||||||
config_host_data.set('CONFIG_LIBPMEM', libpmem.found())
|
|
||||||
config_host_data.set('CONFIG_RBD', rbd.found())
|
|
||||||
+config_host_data.set('CONFIG_VITASTOR', vitastor.found())
|
|
||||||
config_host_data.set('CONFIG_SDL', sdl.found())
|
|
||||||
config_host_data.set('CONFIG_SDL_IMAGE', sdl_image.found())
|
|
||||||
config_host_data.set('CONFIG_SECCOMP', seccomp.found())
|
|
||||||
@@ -3087,6 +3108,7 @@ summary_info += {'bpf support': libbpf.f
|
|
||||||
# TODO: add back protocol and server version
|
|
||||||
summary_info += {'spice support': config_host.has_key('CONFIG_SPICE')}
|
|
||||||
summary_info += {'rbd support': rbd.found()}
|
|
||||||
+summary_info += {'vitastor support': vitastor.found()}
|
|
||||||
summary_info += {'xfsctl support': config_host.has_key('CONFIG_XFS')}
|
|
||||||
summary_info += {'smartcard support': cacard.found()}
|
|
||||||
summary_info += {'U2F support': u2f.found()}
|
|
||||||
Index: pve-qemu-kvm-6.1.0/meson_options.txt
|
|
||||||
===================================================================
|
|
||||||
--- pve-qemu-kvm-6.1.0.orig/meson_options.txt
|
|
||||||
+++ pve-qemu-kvm-6.1.0/meson_options.txt
|
|
||||||
@@ -102,6 +102,8 @@ option('lzo', type : 'feature', value :
|
|
||||||
description: 'lzo compression support')
|
|
||||||
option('rbd', type : 'feature', value : 'auto',
|
|
||||||
description: 'Ceph block device driver')
|
|
||||||
+option('vitastor', type : 'feature', value : 'auto',
|
|
||||||
+ description: 'Vitastor block device driver')
|
|
||||||
option('gtk', type : 'feature', value : 'auto',
|
|
||||||
description: 'GTK+ user interface')
|
|
||||||
option('sdl', type : 'feature', value : 'auto',
|
|
|
@ -1,176 +0,0 @@
|
||||||
diff --git a/block/Makefile.objs b/block/Makefile.objs
|
|
||||||
index 46d585cfd0..62222f25fe 100644
|
|
||||||
--- a/block/Makefile.objs
|
|
||||||
+++ b/block/Makefile.objs
|
|
||||||
@@ -29,6 +29,7 @@ block-obj-$(if $(CONFIG_LIBISCSI),y,n) += iscsi-opts.o
|
|
||||||
block-obj-$(CONFIG_LIBNFS) += nfs.o
|
|
||||||
block-obj-$(CONFIG_CURL) += curl.o
|
|
||||||
block-obj-$(CONFIG_RBD) += rbd.o
|
|
||||||
+block-obj-$(CONFIG_VITASTOR) += vitastor.o
|
|
||||||
block-obj-$(CONFIG_GLUSTERFS) += gluster.o
|
|
||||||
block-obj-$(CONFIG_VXHS) += vxhs.o
|
|
||||||
block-obj-$(CONFIG_LIBSSH2) += ssh.o
|
|
||||||
@@ -49,6 +50,8 @@ curl.o-cflags := $(CURL_CFLAGS)
|
|
||||||
curl.o-libs := $(CURL_LIBS)
|
|
||||||
rbd.o-cflags := $(RBD_CFLAGS)
|
|
||||||
rbd.o-libs := $(RBD_LIBS)
|
|
||||||
+vitastor.o-cflags := $(VITASTOR_CFLAGS)
|
|
||||||
+vitastor.o-libs := $(VITASTOR_LIBS)
|
|
||||||
gluster.o-cflags := $(GLUSTERFS_CFLAGS)
|
|
||||||
gluster.o-libs := $(GLUSTERFS_LIBS)
|
|
||||||
vxhs.o-libs := $(VXHS_LIBS)
|
|
||||||
diff --git a/configure b/configure
|
|
||||||
index 1c9f6090e8..25ef89c33a 100755
|
|
||||||
--- a/configure
|
|
||||||
+++ b/configure
|
|
||||||
@@ -422,6 +422,7 @@ trace_backends="log"
|
|
||||||
trace_file="trace"
|
|
||||||
spice=""
|
|
||||||
rbd=""
|
|
||||||
+vitastor=""
|
|
||||||
smartcard=""
|
|
||||||
libusb=""
|
|
||||||
usb_redir=""
|
|
||||||
@@ -1282,6 +1283,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="yes"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="no"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="yes"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1737,6 +1742,7 @@ disabled with --disable-FEATURE, default is enabled if available:
|
|
||||||
vhost-crypto vhost-crypto acceleration support
|
|
||||||
spice spice
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -3722,6 +3728,27 @@ EOF
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
+##########################################
|
|
||||||
+# vitastor probe
|
|
||||||
+if test "$vitastor" != "no" ; then
|
|
||||||
+ cat > $TMPC <<EOF
|
|
||||||
+#include <vitastor_c.h>
|
|
||||||
+int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+EOF
|
|
||||||
+ vitastor_libs="-lvitastor_client"
|
|
||||||
+ if compile_prog "" "$vitastor_libs" ; then
|
|
||||||
+ vitastor=yes
|
|
||||||
+ else
|
|
||||||
+ if test "$vitastor" = "yes" ; then
|
|
||||||
+ feature_not_found "vitastor block device" "Install vitastor-client-dev"
|
|
||||||
+ fi
|
|
||||||
+ vitastor=no
|
|
||||||
+ fi
|
|
||||||
+fi
|
|
||||||
+
|
|
||||||
##########################################
|
|
||||||
# libssh2 probe
|
|
||||||
min_libssh2_version=1.2.8
|
|
||||||
@@ -6109,6 +6136,7 @@ echo "Trace output file $trace_file-<pid>"
|
|
||||||
fi
|
|
||||||
echo "spice support $spice $(echo_version $spice $spice_protocol_version/$spice_server_version)"
|
|
||||||
echo "rbd support $rbd"
|
|
||||||
+echo "vitastor support $vitastor"
|
|
||||||
echo "xfsctl support $xfs"
|
|
||||||
echo "smartcard support $smartcard"
|
|
||||||
echo "libusb $libusb"
|
|
||||||
@@ -6694,6 +6722,11 @@ if test "$rbd" = "yes" ; then
|
|
||||||
echo "RBD_CFLAGS=$rbd_cflags" >> $config_host_mak
|
|
||||||
echo "RBD_LIBS=$rbd_libs" >> $config_host_mak
|
|
||||||
fi
|
|
||||||
+if test "$vitastor" = "yes" ; then
|
|
||||||
+ echo "CONFIG_VITASTOR=m" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_CFLAGS=$vitastor_cflags" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_LIBS=$vitastor_libs" >> $config_host_mak
|
|
||||||
+fi
|
|
||||||
|
|
||||||
echo "CONFIG_COROUTINE_BACKEND=$coroutine" >> $config_host_mak
|
|
||||||
if test "$coroutine_pool" = "yes" ; then
|
|
||||||
diff --git a/qapi/block-core.json b/qapi/block-core.json
|
|
||||||
index d4fe710836..dbad3327b3 100644
|
|
||||||
--- a/qapi/block-core.json
|
|
||||||
+++ b/qapi/block-core.json
|
|
||||||
@@ -2617,7 +2617,7 @@
|
|
||||||
##
|
|
||||||
{ 'enum': 'BlockdevDriver',
|
|
||||||
'data': [ 'blkdebug', 'blklogwrites', 'blkverify', 'bochs', 'cloop',
|
|
||||||
- 'copy-on-read', 'dmg', 'file', 'ftp', 'ftps', 'gluster',
|
|
||||||
+ 'copy-on-read', 'dmg', 'file', 'ftp', 'ftps', 'gluster', 'vitastor',
|
|
||||||
'host_cdrom', 'host_device', 'http', 'https', 'iscsi', 'luks',
|
|
||||||
'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels', 'qcow',
|
|
||||||
'qcow2', 'qed', 'quorum', 'raw', 'rbd', 'replication', 'sheepdog',
|
|
||||||
@@ -3366,6 +3366,28 @@
|
|
||||||
'*snap-id': 'uint32',
|
|
||||||
'*tag': 'str' } }
|
|
||||||
|
|
||||||
+##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
@@ -3713,6 +3735,7 @@
|
|
||||||
'rbd': 'BlockdevOptionsRbd',
|
|
||||||
'replication':'BlockdevOptionsReplication',
|
|
||||||
'sheepdog': 'BlockdevOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevOptionsSsh',
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
@@ -4157,6 +4180,17 @@
|
|
||||||
'*subformat': 'BlockdevVhdxSubformat',
|
|
||||||
'*block-state-zero': 'bool' } }
|
|
||||||
|
|
||||||
+##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
##
|
|
||||||
# @BlockdevVpcSubformat:
|
|
||||||
#
|
|
||||||
@@ -4212,6 +4246,7 @@
|
|
||||||
'qed': 'BlockdevCreateOptionsQed',
|
|
||||||
'rbd': 'BlockdevCreateOptionsRbd',
|
|
||||||
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
|
@ -1,173 +0,0 @@
|
||||||
diff -NaurpbB qemu-4.2.0/block/Makefile.objs qemu-4.2.0-vitastor/block/Makefile.objs
|
|
||||||
--- qemu-4.2.0/block/Makefile.objs 2019-12-12 18:20:47.000000000 +0000
|
|
||||||
+++ qemu-4.2.0-vitastor/block/Makefile.objs 2021-12-01 21:28:47.342341760 +0000
|
|
||||||
@@ -29,6 +29,7 @@ block-obj-$(if $(CONFIG_LIBISCSI),y,n) +
|
|
||||||
block-obj-$(CONFIG_LIBNFS) += nfs.o
|
|
||||||
block-obj-$(CONFIG_CURL) += curl.o
|
|
||||||
block-obj-$(CONFIG_RBD) += rbd.o
|
|
||||||
+block-obj-$(CONFIG_VITASTOR) += vitastor.o
|
|
||||||
block-obj-$(CONFIG_GLUSTERFS) += gluster.o
|
|
||||||
block-obj-$(CONFIG_VXHS) += vxhs.o
|
|
||||||
block-obj-$(CONFIG_LIBSSH) += ssh.o
|
|
||||||
@@ -53,6 +54,8 @@ curl.o-cflags := $(CURL_CFLAGS)
|
|
||||||
curl.o-libs := $(CURL_LIBS)
|
|
||||||
rbd.o-cflags := $(RBD_CFLAGS)
|
|
||||||
rbd.o-libs := $(RBD_LIBS)
|
|
||||||
+vitastor.o-cflags := $(VITASTOR_CFLAGS)
|
|
||||||
+vitastor.o-libs := $(VITASTOR_LIBS)
|
|
||||||
gluster.o-cflags := $(GLUSTERFS_CFLAGS)
|
|
||||||
gluster.o-libs := $(GLUSTERFS_LIBS)
|
|
||||||
vxhs.o-libs := $(VXHS_LIBS)
|
|
||||||
diff -NaurpbB qemu-4.2.0/configure qemu-4.2.0-vitastor/configure
|
|
||||||
--- qemu-4.2.0/configure 2019-12-12 18:20:47.000000000 +0000
|
|
||||||
+++ qemu-4.2.0-vitastor/configure 2021-12-01 21:30:24.605237562 +0000
|
|
||||||
@@ -435,6 +435,7 @@ trace_backends="log"
|
|
||||||
trace_file="trace"
|
|
||||||
spice=""
|
|
||||||
rbd=""
|
|
||||||
+vitastor=""
|
|
||||||
smartcard=""
|
|
||||||
libusb=""
|
|
||||||
usb_redir=""
|
|
||||||
@@ -1312,6 +1313,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="yes"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="no"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="yes"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1782,6 +1787,7 @@ disabled with --disable-FEATURE, default
|
|
||||||
vhost-user vhost-user backend support
|
|
||||||
spice spice
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -3980,6 +3986,27 @@ EOF
|
|
||||||
fi
|
|
||||||
|
|
||||||
##########################################
|
|
||||||
+# vitastor probe
|
|
||||||
+if test "$vitastor" != "no" ; then
|
|
||||||
+ cat > $TMPC <<EOF
|
|
||||||
+#include <vitastor_c.h>
|
|
||||||
+int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+EOF
|
|
||||||
+ vitastor_libs="-lvitastor_client"
|
|
||||||
+ if compile_prog "" "$vitastor_libs" ; then
|
|
||||||
+ vitastor=yes
|
|
||||||
+ else
|
|
||||||
+ if test "$vitastor" = "yes" ; then
|
|
||||||
+ feature_not_found "vitastor block device" "Install vitastor-client-dev"
|
|
||||||
+ fi
|
|
||||||
+ vitastor=no
|
|
||||||
+ fi
|
|
||||||
+fi
|
|
||||||
+
|
|
||||||
+##########################################
|
|
||||||
# libssh probe
|
|
||||||
if test "$libssh" != "no" ; then
|
|
||||||
if $pkg_config --exists libssh; then
|
|
||||||
@@ -6549,6 +6576,7 @@ echo "Trace output file $trace_file-<pid
|
|
||||||
fi
|
|
||||||
echo "spice support $spice $(echo_version $spice $spice_protocol_version/$spice_server_version)"
|
|
||||||
echo "rbd support $rbd"
|
|
||||||
+echo "vitastor support $vitastor"
|
|
||||||
echo "xfsctl support $xfs"
|
|
||||||
echo "smartcard support $smartcard"
|
|
||||||
echo "libusb $libusb"
|
|
||||||
@@ -7182,6 +7210,11 @@ if test "$rbd" = "yes" ; then
|
|
||||||
echo "RBD_CFLAGS=$rbd_cflags" >> $config_host_mak
|
|
||||||
echo "RBD_LIBS=$rbd_libs" >> $config_host_mak
|
|
||||||
fi
|
|
||||||
+if test "$vitastor" = "yes" ; then
|
|
||||||
+ echo "CONFIG_VITASTOR=m" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_CFLAGS=$vitastor_cflags" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_LIBS=$vitastor_libs" >> $config_host_mak
|
|
||||||
+fi
|
|
||||||
|
|
||||||
echo "CONFIG_COROUTINE_BACKEND=$coroutine" >> $config_host_mak
|
|
||||||
if test "$coroutine_pool" = "yes" ; then
|
|
||||||
diff -NaurpbB qemu-4.2.0/qapi/block-core.json qemu-4.2.0-vitastor/qapi/block-core.json
|
|
||||||
--- qemu-4.2.0/qapi/block-core.json 2019-12-12 18:20:48.000000000 +0000
|
|
||||||
+++ qemu-4.2.0-vitastor/qapi/block-core.json 2021-12-01 21:27:49.213574396 +0000
|
|
||||||
@@ -2894,7 +2894,7 @@
|
|
||||||
'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels', 'qcow',
|
|
||||||
'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
- 'sheepdog',
|
|
||||||
+ 'sheepdog', 'vitastor',
|
|
||||||
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat', 'vxhs' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
@@ -3712,6 +3712,28 @@
|
|
||||||
'*tag': 'str' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
# An enumeration of replication modes.
|
|
||||||
@@ -4071,6 +4093,7 @@
|
|
||||||
'replication': { 'type': 'BlockdevOptionsReplication',
|
|
||||||
'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
'sheepdog': 'BlockdevOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevOptionsSsh',
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
@@ -4441,6 +4464,17 @@
|
|
||||||
'*cluster-size' : 'size' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
# Subformat options for VMDK images
|
|
||||||
@@ -4702,6 +4736,7 @@
|
|
||||||
'qed': 'BlockdevCreateOptionsQed',
|
|
||||||
'rbd': 'BlockdevCreateOptionsRbd',
|
|
||||||
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
|
@ -1,176 +0,0 @@
|
||||||
diff --git a/block/Makefile.objs b/block/Makefile.objs
|
|
||||||
index 3635b6b4c1..6cdf6df6ff 100644
|
|
||||||
--- a/block/Makefile.objs
|
|
||||||
+++ b/block/Makefile.objs
|
|
||||||
@@ -30,6 +30,7 @@ block-obj-$(if $(CONFIG_LIBISCSI),y,n) += iscsi-opts.o
|
|
||||||
block-obj-$(CONFIG_LIBNFS) += nfs.o
|
|
||||||
block-obj-$(CONFIG_CURL) += curl.o
|
|
||||||
block-obj-$(CONFIG_RBD) += rbd.o
|
|
||||||
+block-obj-$(CONFIG_VITASTOR) += vitastor.o
|
|
||||||
block-obj-$(CONFIG_GLUSTERFS) += gluster.o
|
|
||||||
block-obj-$(CONFIG_VXHS) += vxhs.o
|
|
||||||
block-obj-$(CONFIG_LIBSSH) += ssh.o
|
|
||||||
@@ -58,6 +59,8 @@ curl.o-cflags := $(CURL_CFLAGS)
|
|
||||||
curl.o-libs := $(CURL_LIBS)
|
|
||||||
rbd.o-cflags := $(RBD_CFLAGS)
|
|
||||||
rbd.o-libs := $(RBD_LIBS)
|
|
||||||
+vitastor.o-cflags := $(VITASTOR_CFLAGS)
|
|
||||||
+vitastor.o-libs := $(VITASTOR_LIBS)
|
|
||||||
gluster.o-cflags := $(GLUSTERFS_CFLAGS)
|
|
||||||
gluster.o-libs := $(GLUSTERFS_LIBS)
|
|
||||||
vxhs.o-libs := $(VXHS_LIBS)
|
|
||||||
diff --git a/configure b/configure
|
|
||||||
index 23b5e93752..7400cb9546 100755
|
|
||||||
--- a/configure
|
|
||||||
+++ b/configure
|
|
||||||
@@ -438,6 +438,7 @@ trace_backends="log"
|
|
||||||
trace_file="trace"
|
|
||||||
spice=""
|
|
||||||
rbd=""
|
|
||||||
+vitastor=""
|
|
||||||
smartcard=""
|
|
||||||
libusb=""
|
|
||||||
usb_redir=""
|
|
||||||
@@ -1355,6 +1356,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="yes"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="no"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="yes"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1848,6 +1853,7 @@ disabled with --disable-FEATURE, default is enabled if available:
|
|
||||||
vhost-user vhost-user backend support
|
|
||||||
spice spice
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -4088,6 +4094,27 @@ EOF
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
|
|
||||||
+##########################################
|
|
||||||
+# vitastor probe
|
|
||||||
+if test "$vitastor" != "no" ; then
|
|
||||||
+ cat > $TMPC <<EOF
|
|
||||||
+#include <vitastor_c.h>
|
|
||||||
+int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+EOF
|
|
||||||
+ vitastor_libs="-lvitastor_client"
|
|
||||||
+ if compile_prog "" "$vitastor_libs" ; then
|
|
||||||
+ vitastor=yes
|
|
||||||
+ else
|
|
||||||
+ if test "$vitastor" = "yes" ; then
|
|
||||||
+ feature_not_found "vitastor block device" "Install vitastor-client-dev"
|
|
||||||
+ fi
|
|
||||||
+ vitastor=no
|
|
||||||
+ fi
|
|
||||||
+fi
|
|
||||||
+
|
|
||||||
##########################################
|
|
||||||
# libssh probe
|
|
||||||
if test "$libssh" != "no" ; then
|
|
||||||
@@ -6679,6 +6706,7 @@ echo "Trace output file $trace_file-<pid>"
|
|
||||||
fi
|
|
||||||
echo "spice support $spice $(echo_version $spice $spice_protocol_version/$spice_server_version)"
|
|
||||||
echo "rbd support $rbd"
|
|
||||||
+echo "vitastor support $vitastor"
|
|
||||||
echo "xfsctl support $xfs"
|
|
||||||
echo "smartcard support $smartcard"
|
|
||||||
echo "libusb $libusb"
|
|
||||||
@@ -7329,6 +7357,11 @@ if test "$rbd" = "yes" ; then
|
|
||||||
echo "RBD_CFLAGS=$rbd_cflags" >> $config_host_mak
|
|
||||||
echo "RBD_LIBS=$rbd_libs" >> $config_host_mak
|
|
||||||
fi
|
|
||||||
+if test "$vitastor" = "yes" ; then
|
|
||||||
+ echo "CONFIG_VITASTOR=m" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_CFLAGS=$vitastor_cflags" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_LIBS=$vitastor_libs" >> $config_host_mak
|
|
||||||
+fi
|
|
||||||
|
|
||||||
echo "CONFIG_COROUTINE_BACKEND=$coroutine" >> $config_host_mak
|
|
||||||
if test "$coroutine_pool" = "yes" ; then
|
|
||||||
diff --git a/qapi/block-core.json b/qapi/block-core.json
|
|
||||||
index 943df1926a..c4f23230a3 100644
|
|
||||||
--- a/qapi/block-core.json
|
|
||||||
+++ b/qapi/block-core.json
|
|
||||||
@@ -2798,7 +2798,7 @@
|
|
||||||
'luks', 'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels',
|
|
||||||
'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
- 'sheepdog',
|
|
||||||
+ 'sheepdog', 'vitastor',
|
|
||||||
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat', 'vxhs' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
@@ -3634,6 +3634,28 @@
|
|
||||||
'*snap-id': 'uint32',
|
|
||||||
'*tag': 'str' } }
|
|
||||||
|
|
||||||
+##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
@@ -3995,6 +4017,7 @@
|
|
||||||
'replication': { 'type': 'BlockdevOptionsReplication',
|
|
||||||
'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
'sheepdog': 'BlockdevOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevOptionsSsh',
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
@@ -4364,6 +4387,17 @@
|
|
||||||
'size': 'size',
|
|
||||||
'*cluster-size' : 'size' } }
|
|
||||||
|
|
||||||
+##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
@@ -4626,6 +4660,7 @@
|
|
||||||
'qed': 'BlockdevCreateOptionsQed',
|
|
||||||
'rbd': 'BlockdevCreateOptionsRbd',
|
|
||||||
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
|
@ -1,181 +0,0 @@
|
||||||
Index: qemu-5.2+dfsg/qapi/block-core.json
|
|
||||||
===================================================================
|
|
||||||
--- qemu-5.2+dfsg.orig/qapi/block-core.json
|
|
||||||
+++ qemu-5.2+dfsg/qapi/block-core.json
|
|
||||||
@@ -2831,7 +2831,7 @@
|
|
||||||
'luks', 'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels',
|
|
||||||
'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
- 'sheepdog',
|
|
||||||
+ 'sheepdog', 'vitastor',
|
|
||||||
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
@@ -3668,6 +3668,28 @@
|
|
||||||
'*tag': 'str' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
# An enumeration of replication modes.
|
|
||||||
@@ -4015,6 +4037,7 @@
|
|
||||||
'replication': { 'type': 'BlockdevOptionsReplication',
|
|
||||||
'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
'sheepdog': 'BlockdevOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevOptionsSsh',
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
@@ -4404,6 +4427,17 @@
|
|
||||||
'*cluster-size' : 'size' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
# Subformat options for VMDK images
|
|
||||||
@@ -4665,6 +4699,7 @@
|
|
||||||
'qed': 'BlockdevCreateOptionsQed',
|
|
||||||
'rbd': 'BlockdevCreateOptionsRbd',
|
|
||||||
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
||||||
Index: qemu-5.2+dfsg/block/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- qemu-5.2+dfsg.orig/block/meson.build
|
|
||||||
+++ qemu-5.2+dfsg/block/meson.build
|
|
||||||
@@ -76,6 +76,7 @@ foreach m : [
|
|
||||||
['CONFIG_LIBNFS', 'nfs', libnfs, 'nfs.c'],
|
|
||||||
['CONFIG_LIBSSH', 'ssh', libssh, 'ssh.c'],
|
|
||||||
['CONFIG_RBD', 'rbd', rbd, 'rbd.c'],
|
|
||||||
+ ['CONFIG_VITASTOR', 'vitastor', vitastor, 'vitastor.c'],
|
|
||||||
]
|
|
||||||
if config_host.has_key(m[0])
|
|
||||||
if enable_modules
|
|
||||||
Index: qemu-5.2+dfsg/configure
|
|
||||||
===================================================================
|
|
||||||
--- qemu-5.2+dfsg.orig/configure
|
|
||||||
+++ qemu-5.2+dfsg/configure
|
|
||||||
@@ -372,6 +372,7 @@ trace_backends="log"
|
|
||||||
trace_file="trace"
|
|
||||||
spice=""
|
|
||||||
rbd=""
|
|
||||||
+vitastor=""
|
|
||||||
smartcard=""
|
|
||||||
u2f="auto"
|
|
||||||
libusb=""
|
|
||||||
@@ -1263,6 +1264,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="yes"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="no"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="yes"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1827,6 +1832,7 @@ disabled with --disable-FEATURE, default
|
|
||||||
vhost-vdpa vhost-vdpa kernel backend support
|
|
||||||
spice spice
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -3719,6 +3725,27 @@ EOF
|
|
||||||
fi
|
|
||||||
|
|
||||||
##########################################
|
|
||||||
+# vitastor probe
|
|
||||||
+if test "$vitastor" != "no" ; then
|
|
||||||
+ cat > $TMPC <<EOF
|
|
||||||
+#include <vitastor_c.h>
|
|
||||||
+int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+}
|
|
||||||
+EOF
|
|
||||||
+ vitastor_libs="-lvitastor_client"
|
|
||||||
+ if compile_prog "" "$vitastor_libs" ; then
|
|
||||||
+ vitastor=yes
|
|
||||||
+ else
|
|
||||||
+ if test "$vitastor" = "yes" ; then
|
|
||||||
+ feature_not_found "vitastor block device" "Install vitastor-client-dev"
|
|
||||||
+ fi
|
|
||||||
+ vitastor=no
|
|
||||||
+ fi
|
|
||||||
+fi
|
|
||||||
+
|
|
||||||
+##########################################
|
|
||||||
# libssh probe
|
|
||||||
if test "$libssh" != "no" ; then
|
|
||||||
if $pkg_config --exists libssh; then
|
|
||||||
@@ -6456,6 +6483,10 @@ if test "$rbd" = "yes" ; then
|
|
||||||
echo "CONFIG_RBD=y" >> $config_host_mak
|
|
||||||
echo "RBD_LIBS=$rbd_libs" >> $config_host_mak
|
|
||||||
fi
|
|
||||||
+if test "$vitastor" = "yes" ; then
|
|
||||||
+ echo "CONFIG_VITASTOR=y" >> $config_host_mak
|
|
||||||
+ echo "VITASTOR_LIBS=$vitastor_libs" >> $config_host_mak
|
|
||||||
+fi
|
|
||||||
|
|
||||||
echo "CONFIG_COROUTINE_BACKEND=$coroutine" >> $config_host_mak
|
|
||||||
if test "$coroutine_pool" = "yes" ; then
|
|
||||||
Index: qemu-5.2+dfsg/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- qemu-5.2+dfsg.orig/meson.build
|
|
||||||
+++ qemu-5.2+dfsg/meson.build
|
|
||||||
@@ -596,6 +596,10 @@ rbd = not_found
|
|
||||||
if 'CONFIG_RBD' in config_host
|
|
||||||
rbd = declare_dependency(link_args: config_host['RBD_LIBS'].split())
|
|
||||||
endif
|
|
||||||
+vitastor = not_found
|
|
||||||
+if 'CONFIG_VITASTOR' in config_host
|
|
||||||
+ vitastor = declare_dependency(link_args: config_host['VITASTOR_LIBS'].split())
|
|
||||||
+endif
|
|
||||||
glusterfs = not_found
|
|
||||||
if 'CONFIG_GLUSTERFS' in config_host
|
|
||||||
glusterfs = declare_dependency(compile_args: config_host['GLUSTERFS_CFLAGS'].split(),
|
|
||||||
@@ -2145,6 +2149,7 @@ endif
|
|
||||||
# TODO: add back protocol and server version
|
|
||||||
summary_info += {'spice support': config_host.has_key('CONFIG_SPICE')}
|
|
||||||
summary_info += {'rbd support': config_host.has_key('CONFIG_RBD')}
|
|
||||||
+summary_info += {'vitastor support': config_host.has_key('CONFIG_VITASTOR')}
|
|
||||||
summary_info += {'xfsctl support': config_host.has_key('CONFIG_XFS')}
|
|
||||||
summary_info += {'smartcard support': config_host.has_key('CONFIG_SMARTCARD')}
|
|
||||||
summary_info += {'U2F support': u2f.found()}
|
|
|
@ -1,188 +0,0 @@
|
||||||
diff --git a/block/meson.build b/block/meson.build
|
|
||||||
index d21990ec95..385ac37732 100644
|
|
||||||
--- a/block/meson.build
|
|
||||||
+++ b/block/meson.build
|
|
||||||
@@ -78,6 +78,7 @@ foreach m : [
|
|
||||||
[libnfs, 'nfs', libnfs, 'nfs.c'],
|
|
||||||
[libssh, 'ssh', libssh, 'ssh.c'],
|
|
||||||
[rbd, 'rbd', rbd, 'rbd.c'],
|
|
||||||
+ [vitastor, 'vitastor', vitastor, 'vitastor.c'],
|
|
||||||
]
|
|
||||||
if m[0].found()
|
|
||||||
if enable_modules
|
|
||||||
diff --git a/configure b/configure
|
|
||||||
index c77f7b1020..5f534e8484 100755
|
|
||||||
--- a/configure
|
|
||||||
+++ b/configure
|
|
||||||
@@ -389,6 +389,7 @@ trace_backends="log"
|
|
||||||
trace_file="trace"
|
|
||||||
spice="$default_feature"
|
|
||||||
rbd="auto"
|
|
||||||
+vitastor="auto"
|
|
||||||
smartcard="$default_feature"
|
|
||||||
u2f="auto"
|
|
||||||
libusb="$default_feature"
|
|
||||||
@@ -1280,6 +1281,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="enabled"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="disabled"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="enabled"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1867,6 +1872,7 @@ disabled with --disable-FEATURE, default is enabled if available
|
|
||||||
vhost-vdpa vhost-vdpa kernel backend support
|
|
||||||
spice spice
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -6423,7 +6429,7 @@ NINJA=$ninja $meson setup \
|
|
||||||
-Dcapstone=$capstone -Dslirp=$slirp -Dfdt=$fdt -Dbrlapi=$brlapi \
|
|
||||||
-Dcurl=$curl -Dglusterfs=$glusterfs -Dbzip2=$bzip2 -Dlibiscsi=$libiscsi \
|
|
||||||
-Dlibnfs=$libnfs -Diconv=$iconv -Dcurses=$curses -Dlibudev=$libudev\
|
|
||||||
- -Drbd=$rbd -Dlzo=$lzo -Dsnappy=$snappy -Dlzfse=$lzfse \
|
|
||||||
+ -Drbd=$rbd -Dvitastor=$vitastor -Dlzo=$lzo -Dsnappy=$snappy -Dlzfse=$lzfse \
|
|
||||||
-Dzstd=$zstd -Dseccomp=$seccomp -Dvirtfs=$virtfs -Dcap_ng=$cap_ng \
|
|
||||||
-Dattr=$attr -Ddefault_devices=$default_devices \
|
|
||||||
-Ddocs=$docs -Dsphinx_build=$sphinx_build -Dinstall_blobs=$blobs \
|
|
||||||
diff --git a/meson.build b/meson.build
|
|
||||||
index c6f4b0cf5e..3dc7f7b463 100644
|
|
||||||
--- a/meson.build
|
|
||||||
+++ b/meson.build
|
|
||||||
@@ -720,6 +720,26 @@ if not get_option('rbd').auto() or have_block
|
|
||||||
endif
|
|
||||||
endif
|
|
||||||
|
|
||||||
+vitastor = not_found
|
|
||||||
+if not get_option('vitastor').auto() or have_block
|
|
||||||
+ libvitastor_client = cc.find_library('vitastor_client', has_headers: ['vitastor_c.h'],
|
|
||||||
+ required: get_option('vitastor'), kwargs: static_kwargs)
|
|
||||||
+ if libvitastor_client.found()
|
|
||||||
+ if cc.links('''
|
|
||||||
+ #include <vitastor_c.h>
|
|
||||||
+ int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+ }''', dependencies: libvitastor_client)
|
|
||||||
+ vitastor = declare_dependency(dependencies: libvitastor_client)
|
|
||||||
+ elif get_option('vitastor').enabled()
|
|
||||||
+ error('could not link libvitastor_client')
|
|
||||||
+ else
|
|
||||||
+ warning('could not link libvitastor_client, disabling')
|
|
||||||
+ endif
|
|
||||||
+ endif
|
|
||||||
+endif
|
|
||||||
+
|
|
||||||
glusterfs = not_found
|
|
||||||
glusterfs_ftruncate_has_stat = false
|
|
||||||
glusterfs_iocb_has_stat = false
|
|
||||||
@@ -1118,6 +1138,7 @@ config_host_data.set('CONFIG_LIBCAP_NG', libcap_ng.found())
|
|
||||||
config_host_data.set('CONFIG_LIBISCSI', libiscsi.found())
|
|
||||||
config_host_data.set('CONFIG_LIBNFS', libnfs.found())
|
|
||||||
config_host_data.set('CONFIG_RBD', rbd.found())
|
|
||||||
+config_host_data.set('CONFIG_VITASTOR', vitastor.found())
|
|
||||||
config_host_data.set('CONFIG_SDL', sdl.found())
|
|
||||||
config_host_data.set('CONFIG_SDL_IMAGE', sdl_image.found())
|
|
||||||
config_host_data.set('CONFIG_SECCOMP', seccomp.found())
|
|
||||||
@@ -2683,6 +2704,7 @@ summary_info += {'libcap-ng support': libcap_ng.found()}
|
|
||||||
# TODO: add back protocol and server version
|
|
||||||
summary_info += {'spice support': config_host.has_key('CONFIG_SPICE')}
|
|
||||||
summary_info += {'rbd support': rbd.found()}
|
|
||||||
+summary_info += {'vitastor support': vitastor.found()}
|
|
||||||
summary_info += {'xfsctl support': config_host.has_key('CONFIG_XFS')}
|
|
||||||
summary_info += {'smartcard support': config_host.has_key('CONFIG_SMARTCARD')}
|
|
||||||
summary_info += {'U2F support': u2f.found()}
|
|
||||||
diff --git a/meson_options.txt b/meson_options.txt
|
|
||||||
index 9734019995..bc93963b27 100644
|
|
||||||
--- a/meson_options.txt
|
|
||||||
+++ b/meson_options.txt
|
|
||||||
@@ -80,6 +80,8 @@ option('lzo', type : 'feature', value : 'auto',
|
|
||||||
description: 'lzo compression support')
|
|
||||||
option('rbd', type : 'feature', value : 'auto',
|
|
||||||
description: 'Ceph block device driver')
|
|
||||||
+option('vitastor', type : 'feature', value : 'auto',
|
|
||||||
+ description: 'Vitastor block device driver')
|
|
||||||
option('gtk', type : 'feature', value : 'auto',
|
|
||||||
description: 'GTK+ user interface')
|
|
||||||
option('sdl', type : 'feature', value : 'auto',
|
|
||||||
diff --git a/qapi/block-core.json b/qapi/block-core.json
|
|
||||||
index 6d227924d0..d14b29aa43 100644
|
|
||||||
--- a/qapi/block-core.json
|
|
||||||
+++ b/qapi/block-core.json
|
|
||||||
@@ -2819,7 +2819,7 @@
|
|
||||||
'preallocate', 'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
'sheepdog',
|
|
||||||
- 'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
+ 'ssh', 'throttle', 'vdi', 'vhdx', 'vitastor', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
# @BlockdevOptionsFile:
|
|
||||||
@@ -3671,6 +3671,28 @@
|
|
||||||
'*snap-id': 'uint32',
|
|
||||||
'*tag': 'str' } }
|
|
||||||
|
|
||||||
+##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
@@ -4042,6 +4064,7 @@
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
'vhdx': 'BlockdevOptionsGenericFormat',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'vmdk': 'BlockdevOptionsGenericCOWFormat',
|
|
||||||
'vpc': 'BlockdevOptionsGenericFormat',
|
|
||||||
'vvfat': 'BlockdevOptionsVVFAT'
|
|
||||||
@@ -4426,6 +4449,17 @@
|
|
||||||
'size': 'size',
|
|
||||||
'*cluster-size' : 'size' } }
|
|
||||||
|
|
||||||
+##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
@@ -4691,6 +4725,7 @@
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'vmdk': 'BlockdevCreateOptionsVmdk',
|
|
||||||
'vpc': 'BlockdevCreateOptionsVpc'
|
|
||||||
} }
|
|
|
@ -1,188 +0,0 @@
|
||||||
Index: qemu-6.1+dfsg/qapi/block-core.json
|
|
||||||
===================================================================
|
|
||||||
--- qemu-6.1+dfsg.orig/qapi/block-core.json
|
|
||||||
+++ qemu-6.1+dfsg/qapi/block-core.json
|
|
||||||
@@ -2838,7 +2838,7 @@
|
|
||||||
'luks', 'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels',
|
|
||||||
'preallocate', 'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
|
||||||
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
|
||||||
- 'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
+ 'ssh', 'throttle', 'vdi', 'vhdx', 'vitastor', 'vmdk', 'vpc', 'vvfat' ] }
|
|
||||||
|
|
||||||
##
|
|
||||||
# @BlockdevOptionsFile:
|
|
||||||
@@ -3763,6 +3763,28 @@
|
|
||||||
'*server': ['InetSocketAddressBase'] } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific block device options for vitastor
|
|
||||||
+#
|
|
||||||
+# @image: Image name
|
|
||||||
+# @inode: Inode number
|
|
||||||
+# @pool: Pool ID
|
|
||||||
+# @size: Desired image size in bytes
|
|
||||||
+# @config-path: Path to Vitastor configuration
|
|
||||||
+# @etcd-host: etcd connection address(es)
|
|
||||||
+# @etcd-prefix: etcd key/value prefix
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'data': { '*inode': 'uint64',
|
|
||||||
+ '*pool': 'uint64',
|
|
||||||
+ '*size': 'uint64',
|
|
||||||
+ '*image': 'str',
|
|
||||||
+ '*config-path': 'str',
|
|
||||||
+ '*etcd-host': 'str',
|
|
||||||
+ '*etcd-prefix': 'str' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @ReplicationMode:
|
|
||||||
#
|
|
||||||
# An enumeration of replication modes.
|
|
||||||
@@ -4134,6 +4156,7 @@
|
|
||||||
'throttle': 'BlockdevOptionsThrottle',
|
|
||||||
'vdi': 'BlockdevOptionsGenericFormat',
|
|
||||||
'vhdx': 'BlockdevOptionsGenericFormat',
|
|
||||||
+ 'vitastor': 'BlockdevOptionsVitastor',
|
|
||||||
'vmdk': 'BlockdevOptionsGenericCOWFormat',
|
|
||||||
'vpc': 'BlockdevOptionsGenericFormat',
|
|
||||||
'vvfat': 'BlockdevOptionsVVFAT'
|
|
||||||
@@ -4523,6 +4546,17 @@
|
|
||||||
'*encrypt' : 'RbdEncryptionCreateOptions' } }
|
|
||||||
|
|
||||||
##
|
|
||||||
+# @BlockdevCreateOptionsVitastor:
|
|
||||||
+#
|
|
||||||
+# Driver specific image creation options for Vitastor.
|
|
||||||
+#
|
|
||||||
+# @size: Size of the virtual disk in bytes
|
|
||||||
+##
|
|
||||||
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
|
||||||
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
|
||||||
+ 'size': 'size' } }
|
|
||||||
+
|
|
||||||
+##
|
|
||||||
# @BlockdevVmdkSubformat:
|
|
||||||
#
|
|
||||||
# Subformat options for VMDK images
|
|
||||||
@@ -4718,6 +4752,7 @@
|
|
||||||
'ssh': 'BlockdevCreateOptionsSsh',
|
|
||||||
'vdi': 'BlockdevCreateOptionsVdi',
|
|
||||||
'vhdx': 'BlockdevCreateOptionsVhdx',
|
|
||||||
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
|
||||||
'vmdk': 'BlockdevCreateOptionsVmdk',
|
|
||||||
'vpc': 'BlockdevCreateOptionsVpc'
|
|
||||||
} }
|
|
||||||
Index: qemu-6.1+dfsg/block/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- qemu-6.1+dfsg.orig/block/meson.build
|
|
||||||
+++ qemu-6.1+dfsg/block/meson.build
|
|
||||||
@@ -78,6 +78,7 @@ foreach m : [
|
|
||||||
[libnfs, 'nfs', files('nfs.c')],
|
|
||||||
[libssh, 'ssh', files('ssh.c')],
|
|
||||||
[rbd, 'rbd', files('rbd.c')],
|
|
||||||
+ [vitastor, 'vitastor', files('vitastor.c')],
|
|
||||||
]
|
|
||||||
if m[0].found()
|
|
||||||
module_ss = ss.source_set()
|
|
||||||
Index: qemu-6.1+dfsg/configure
|
|
||||||
===================================================================
|
|
||||||
--- qemu-6.1+dfsg.orig/configure
|
|
||||||
+++ qemu-6.1+dfsg/configure
|
|
||||||
@@ -375,6 +375,7 @@ trace_file="trace"
|
|
||||||
spice="$default_feature"
|
|
||||||
spice_protocol="auto"
|
|
||||||
rbd="auto"
|
|
||||||
+vitastor="auto"
|
|
||||||
smartcard="auto"
|
|
||||||
u2f="auto"
|
|
||||||
libusb="auto"
|
|
||||||
@@ -1292,6 +1293,10 @@ for opt do
|
|
||||||
;;
|
|
||||||
--enable-rbd) rbd="enabled"
|
|
||||||
;;
|
|
||||||
+ --disable-vitastor) vitastor="disabled"
|
|
||||||
+ ;;
|
|
||||||
+ --enable-vitastor) vitastor="enabled"
|
|
||||||
+ ;;
|
|
||||||
--disable-xfsctl) xfs="no"
|
|
||||||
;;
|
|
||||||
--enable-xfsctl) xfs="yes"
|
|
||||||
@@ -1916,6 +1921,7 @@ disabled with --disable-FEATURE, default
|
|
||||||
spice spice
|
|
||||||
spice-protocol spice-protocol
|
|
||||||
rbd rados block device (rbd)
|
|
||||||
+ vitastor vitastor block device
|
|
||||||
libiscsi iscsi support
|
|
||||||
libnfs nfs support
|
|
||||||
smartcard smartcard support (libcacard)
|
|
||||||
@@ -5202,7 +5208,7 @@ if test "$skip_meson" = no; then
|
|
||||||
-Dcapstone=$capstone -Dslirp=$slirp -Dfdt=$fdt -Dbrlapi=$brlapi \
|
|
||||||
-Dcurl=$curl -Dglusterfs=$glusterfs -Dbzip2=$bzip2 -Dlibiscsi=$libiscsi \
|
|
||||||
-Dlibnfs=$libnfs -Diconv=$iconv -Dcurses=$curses -Dlibudev=$libudev\
|
|
||||||
- -Drbd=$rbd -Dlzo=$lzo -Dsnappy=$snappy -Dlzfse=$lzfse -Dlibxml2=$libxml2 \
|
|
||||||
+ -Drbd=$rbd -Dvitastor=$vitastor -Dlzo=$lzo -Dsnappy=$snappy -Dlzfse=$lzfse -Dlibxml2=$libxml2 \
|
|
||||||
-Dlibdaxctl=$libdaxctl -Dlibpmem=$libpmem -Dlinux_io_uring=$linux_io_uring \
|
|
||||||
-Dgnutls=$gnutls -Dnettle=$nettle -Dgcrypt=$gcrypt -Dauth_pam=$auth_pam \
|
|
||||||
-Dzstd=$zstd -Dseccomp=$seccomp -Dvirtfs=$virtfs -Dcap_ng=$cap_ng \
|
|
||||||
Index: qemu-6.1+dfsg/meson.build
|
|
||||||
===================================================================
|
|
||||||
--- qemu-6.1+dfsg.orig/meson.build
|
|
||||||
+++ qemu-6.1+dfsg/meson.build
|
|
||||||
@@ -729,6 +729,26 @@ if not get_option('rbd').auto() or have_
|
|
||||||
endif
|
|
||||||
endif
|
|
||||||
|
|
||||||
+vitastor = not_found
|
|
||||||
+if not get_option('vitastor').auto() or have_block
|
|
||||||
+ libvitastor_client = cc.find_library('vitastor_client', has_headers: ['vitastor_c.h'],
|
|
||||||
+ required: get_option('vitastor'), kwargs: static_kwargs)
|
|
||||||
+ if libvitastor_client.found()
|
|
||||||
+ if cc.links('''
|
|
||||||
+ #include <vitastor_c.h>
|
|
||||||
+ int main(void) {
|
|
||||||
+ vitastor_c_create_qemu(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0);
|
|
||||||
+ return 0;
|
|
||||||
+ }''', dependencies: libvitastor_client)
|
|
||||||
+ vitastor = declare_dependency(dependencies: libvitastor_client)
|
|
||||||
+ elif get_option('vitastor').enabled()
|
|
||||||
+ error('could not link libvitastor_client')
|
|
||||||
+ else
|
|
||||||
+ warning('could not link libvitastor_client, disabling')
|
|
||||||
+ endif
|
|
||||||
+ endif
|
|
||||||
+endif
|
|
||||||
+
|
|
||||||
glusterfs = not_found
|
|
||||||
glusterfs_ftruncate_has_stat = false
|
|
||||||
glusterfs_iocb_has_stat = false
|
|
||||||
@@ -1264,6 +1284,7 @@ config_host_data.set('CONFIG_LIBNFS', li
|
|
||||||
config_host_data.set('CONFIG_LINUX_IO_URING', linux_io_uring.found())
|
|
||||||
config_host_data.set('CONFIG_LIBPMEM', libpmem.found())
|
|
||||||
config_host_data.set('CONFIG_RBD', rbd.found())
|
|
||||||
+config_host_data.set('CONFIG_VITASTOR', vitastor.found())
|
|
||||||
config_host_data.set('CONFIG_SDL', sdl.found())
|
|
||||||
config_host_data.set('CONFIG_SDL_IMAGE', sdl_image.found())
|
|
||||||
config_host_data.set('CONFIG_SECCOMP', seccomp.found())
|
|
||||||
@@ -3075,6 +3096,7 @@ summary_info += {'bpf support': libbpf.f
|
|
||||||
# TODO: add back protocol and server version
|
|
||||||
summary_info += {'spice support': config_host.has_key('CONFIG_SPICE')}
|
|
||||||
summary_info += {'rbd support': rbd.found()}
|
|
||||||
+summary_info += {'vitastor support': vitastor.found()}
|
|
||||||
summary_info += {'xfsctl support': config_host.has_key('CONFIG_XFS')}
|
|
||||||
summary_info += {'smartcard support': cacard.found()}
|
|
||||||
summary_info += {'U2F support': u2f.found()}
|
|
||||||
Index: qemu-6.1+dfsg/meson_options.txt
|
|
||||||
===================================================================
|
|
||||||
--- qemu-6.1+dfsg.orig/meson_options.txt
|
|
||||||
+++ qemu-6.1+dfsg/meson_options.txt
|
|
||||||
@@ -102,6 +102,8 @@ option('lzo', type : 'feature', value :
|
|
||||||
description: 'lzo compression support')
|
|
||||||
option('rbd', type : 'feature', value : 'auto',
|
|
||||||
description: 'Ceph block device driver')
|
|
||||||
+option('vitastor', type : 'feature', value : 'auto',
|
|
||||||
+ description: 'Vitastor block device driver')
|
|
||||||
option('gtk', type : 'feature', value : 'auto',
|
|
||||||
description: 'GTK+ user interface')
|
|
||||||
option('sdl', type : 'feature', value : 'auto',
|
|
|
@ -1,15 +0,0 @@
|
||||||
#!/bin/bash
|
|
||||||
# QEMU patches don't include the `block/vitastor.c` file to not duplicate it in sources
|
|
||||||
# Run this script to append its creation to all QEMU patches
|
|
||||||
|
|
||||||
DIR=$(dirname $0)
|
|
||||||
for i in "$DIR"/qemu-*-vitastor.patch "$DIR"/pve-qemu-*-vitastor.patch; do
|
|
||||||
if ! grep -qP '^\+\+\+ .*block/vitastor\.c' $i; then
|
|
||||||
echo 'Index: a/block/vitastor.c' >> $i
|
|
||||||
echo '===================================================================' >> $i
|
|
||||||
echo '--- /dev/null' >> $i
|
|
||||||
echo '+++ a/block/vitastor.c' >> $i
|
|
||||||
echo '@@ -0,0 +1,'$(wc -l "$DIR"/../src/qemu_driver.c)' @@' >> $i
|
|
||||||
cat "$DIR"/../src/qemu_driver.c | sed 's/^/+/' >> $i
|
|
||||||
fi
|
|
||||||
done
|
|
|
@ -0,0 +1,84 @@
|
||||||
|
Index: qemu-3.1+dfsg/qapi/block-core.json
|
||||||
|
===================================================================
|
||||||
|
--- qemu-3.1+dfsg.orig/qapi/block-core.json
|
||||||
|
+++ qemu-3.1+dfsg/qapi/block-core.json
|
||||||
|
@@ -2617,7 +2617,7 @@
|
||||||
|
##
|
||||||
|
{ 'enum': 'BlockdevDriver',
|
||||||
|
'data': [ 'blkdebug', 'blklogwrites', 'blkverify', 'bochs', 'cloop',
|
||||||
|
- 'copy-on-read', 'dmg', 'file', 'ftp', 'ftps', 'gluster',
|
||||||
|
+ 'copy-on-read', 'dmg', 'file', 'ftp', 'ftps', 'gluster', 'vitastor',
|
||||||
|
'host_cdrom', 'host_device', 'http', 'https', 'iscsi', 'luks',
|
||||||
|
'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels', 'qcow',
|
||||||
|
'qcow2', 'qed', 'quorum', 'raw', 'rbd', 'replication', 'sheepdog',
|
||||||
|
@@ -3367,6 +3367,24 @@
|
||||||
|
'*tag': 'str' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific block device options for vitastor
|
||||||
|
+#
|
||||||
|
+# @inode: Inode number
|
||||||
|
+# @pool: Pool ID
|
||||||
|
+# @size: Desired image size in bytes
|
||||||
|
+# @etcd_host: etcd connection address
|
||||||
|
+# @etcd_prefix: etcd key/value prefix
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'data': { 'inode': 'uint64',
|
||||||
|
+ 'pool': 'uint64',
|
||||||
|
+ 'size': 'uint64',
|
||||||
|
+ 'etcd_host': 'str',
|
||||||
|
+ '*etcd_prefix': 'str' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @ReplicationMode:
|
||||||
|
#
|
||||||
|
# An enumeration of replication modes.
|
||||||
|
@@ -3713,6 +3731,7 @@
|
||||||
|
'rbd': 'BlockdevOptionsRbd',
|
||||||
|
'replication':'BlockdevOptionsReplication',
|
||||||
|
'sheepdog': 'BlockdevOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevOptionsSsh',
|
||||||
|
'throttle': 'BlockdevOptionsThrottle',
|
||||||
|
'vdi': 'BlockdevOptionsGenericFormat',
|
||||||
|
@@ -4158,6 +4177,17 @@
|
||||||
|
'*block-state-zero': 'bool' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevCreateOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific image creation options for Vitastor.
|
||||||
|
+#
|
||||||
|
+# @size: Size of the virtual disk in bytes
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
||||||
|
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'size': 'size' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @BlockdevVpcSubformat:
|
||||||
|
#
|
||||||
|
# @dynamic: Growing image file
|
||||||
|
@@ -4212,6 +4242,7 @@
|
||||||
|
'qed': 'BlockdevCreateOptionsQed',
|
||||||
|
'rbd': 'BlockdevCreateOptionsRbd',
|
||||||
|
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevCreateOptionsSsh',
|
||||||
|
'vdi': 'BlockdevCreateOptionsVdi',
|
||||||
|
'vhdx': 'BlockdevCreateOptionsVhdx',
|
||||||
|
Index: qemu-3.1+dfsg/scripts/modules/module_block.py
|
||||||
|
===================================================================
|
||||||
|
--- qemu-3.1+dfsg.orig/scripts/modules/module_block.py
|
||||||
|
+++ qemu-3.1+dfsg/scripts/modules/module_block.py
|
||||||
|
@@ -88,6 +88,7 @@ def print_bottom(fheader):
|
||||||
|
output_file = sys.argv[1]
|
||||||
|
with open(output_file, 'w') as fheader:
|
||||||
|
print_top(fheader)
|
||||||
|
+ add_module(fheader, "vitastor", "vitastor", "vitastor")
|
||||||
|
|
||||||
|
for filename in sys.argv[2:]:
|
||||||
|
if os.path.isfile(filename):
|
|
@ -0,0 +1,84 @@
|
||||||
|
Index: qemu/qapi/block-core.json
|
||||||
|
===================================================================
|
||||||
|
--- qemu.orig/qapi/block-core.json 2020-11-07 22:57:38.932613674 +0000
|
||||||
|
+++ qemu.orig/qapi/block-core.json 2020-11-07 22:59:49.890722862 +0000
|
||||||
|
@@ -2907,7 +2907,7 @@
|
||||||
|
'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels', 'qcow',
|
||||||
|
'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
||||||
|
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
||||||
|
- 'sheepdog',
|
||||||
|
+ 'sheepdog', 'vitastor',
|
||||||
|
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat', 'vxhs' ] }
|
||||||
|
|
||||||
|
##
|
||||||
|
@@ -3725,6 +3725,24 @@
|
||||||
|
'*tag': 'str' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific block device options for vitastor
|
||||||
|
+#
|
||||||
|
+# @inode: Inode number
|
||||||
|
+# @pool: Pool ID
|
||||||
|
+# @size: Desired image size in bytes
|
||||||
|
+# @etcd_host: etcd connection address
|
||||||
|
+# @etcd_prefix: etcd key/value prefix
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'data': { 'inode': 'uint64',
|
||||||
|
+ 'pool': 'uint64',
|
||||||
|
+ 'size': 'uint64',
|
||||||
|
+ 'etcd_host': 'str',
|
||||||
|
+ '*etcd_prefix': 'str' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @ReplicationMode:
|
||||||
|
#
|
||||||
|
# An enumeration of replication modes.
|
||||||
|
@@ -4084,6 +4102,7 @@
|
||||||
|
'replication': { 'type': 'BlockdevOptionsReplication',
|
||||||
|
'if': 'defined(CONFIG_REPLICATION)' },
|
||||||
|
'sheepdog': 'BlockdevOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevOptionsSsh',
|
||||||
|
'throttle': 'BlockdevOptionsThrottle',
|
||||||
|
'vdi': 'BlockdevOptionsGenericFormat',
|
||||||
|
@@ -4461,6 +4480,17 @@
|
||||||
|
'*cluster-size' : 'size' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevCreateOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific image creation options for Vitastor.
|
||||||
|
+#
|
||||||
|
+# @size: Size of the virtual disk in bytes
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
||||||
|
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'size': 'size' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @BlockdevVmdkSubformat:
|
||||||
|
#
|
||||||
|
# Subformat options for VMDK images
|
||||||
|
@@ -4722,6 +4752,7 @@
|
||||||
|
'qed': 'BlockdevCreateOptionsQed',
|
||||||
|
'rbd': 'BlockdevCreateOptionsRbd',
|
||||||
|
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevCreateOptionsSsh',
|
||||||
|
'vdi': 'BlockdevCreateOptionsVdi',
|
||||||
|
'vhdx': 'BlockdevCreateOptionsVhdx',
|
||||||
|
Index: qemu/scripts/modules/module_block.py
|
||||||
|
===================================================================
|
||||||
|
--- qemu.orig/scripts/modules/module_block.py 2020-11-07 22:57:38.936613739 +0000
|
||||||
|
+++ qemu/scripts/modules/module_block.py 2020-11-07 22:59:49.890722862 +0000
|
||||||
|
@@ -86,6 +86,7 @@ def print_bottom(fheader):
|
||||||
|
output_file = sys.argv[1]
|
||||||
|
with open(output_file, 'w') as fheader:
|
||||||
|
print_top(fheader)
|
||||||
|
+ add_module(fheader, "vitastor", "vitastor", "vitastor")
|
||||||
|
|
||||||
|
for filename in sys.argv[2:]:
|
||||||
|
if os.path.isfile(filename):
|
|
@ -0,0 +1,84 @@
|
||||||
|
Index: qemu/qapi/block-core.json
|
||||||
|
===================================================================
|
||||||
|
--- qemu.orig/qapi/block-core.json
|
||||||
|
+++ qemu/qapi/block-core.json
|
||||||
|
@@ -2798,7 +2798,7 @@
|
||||||
|
'luks', 'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels',
|
||||||
|
'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
||||||
|
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
||||||
|
- 'sheepdog',
|
||||||
|
+ 'sheepdog', 'vitastor',
|
||||||
|
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat', 'vxhs' ] }
|
||||||
|
|
||||||
|
##
|
||||||
|
@@ -3635,6 +3635,24 @@
|
||||||
|
'*tag': 'str' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific block device options for vitastor
|
||||||
|
+#
|
||||||
|
+# @inode: Inode number
|
||||||
|
+# @pool: Pool ID
|
||||||
|
+# @size: Desired image size in bytes
|
||||||
|
+# @etcd_host: etcd connection address
|
||||||
|
+# @etcd_prefix: etcd key/value prefix
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'data': { 'inode': 'uint64',
|
||||||
|
+ 'pool': 'uint64',
|
||||||
|
+ 'size': 'uint64',
|
||||||
|
+ 'etcd_host': 'str',
|
||||||
|
+ '*etcd_prefix': 'str' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @ReplicationMode:
|
||||||
|
#
|
||||||
|
# An enumeration of replication modes.
|
||||||
|
@@ -3995,6 +4013,7 @@
|
||||||
|
'replication': { 'type': 'BlockdevOptionsReplication',
|
||||||
|
'if': 'defined(CONFIG_REPLICATION)' },
|
||||||
|
'sheepdog': 'BlockdevOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevOptionsSsh',
|
||||||
|
'throttle': 'BlockdevOptionsThrottle',
|
||||||
|
'vdi': 'BlockdevOptionsGenericFormat',
|
||||||
|
@@ -4365,6 +4384,17 @@
|
||||||
|
'*cluster-size' : 'size' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevCreateOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific image creation options for Vitastor.
|
||||||
|
+#
|
||||||
|
+# @size: Size of the virtual disk in bytes
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
||||||
|
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'size': 'size' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @BlockdevVmdkSubformat:
|
||||||
|
#
|
||||||
|
# Subformat options for VMDK images
|
||||||
|
@@ -4626,6 +4656,7 @@
|
||||||
|
'qed': 'BlockdevCreateOptionsQed',
|
||||||
|
'rbd': 'BlockdevCreateOptionsRbd',
|
||||||
|
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevCreateOptionsSsh',
|
||||||
|
'vdi': 'BlockdevCreateOptionsVdi',
|
||||||
|
'vhdx': 'BlockdevCreateOptionsVhdx',
|
||||||
|
Index: qemu/scripts/modules/module_block.py
|
||||||
|
===================================================================
|
||||||
|
--- qemu.orig/scripts/modules/module_block.py
|
||||||
|
+++ qemu/scripts/modules/module_block.py
|
||||||
|
@@ -85,6 +85,7 @@ def print_bottom(fheader):
|
||||||
|
output_file = sys.argv[1]
|
||||||
|
with open(output_file, 'w') as fheader:
|
||||||
|
print_top(fheader)
|
||||||
|
+ add_module(fheader, "vitastor", "vitastor", "vitastor")
|
||||||
|
|
||||||
|
for filename in sys.argv[2:]:
|
||||||
|
if os.path.isfile(filename):
|
|
@ -0,0 +1,84 @@
|
||||||
|
Index: qemu-5.1+dfsg/qapi/block-core.json
|
||||||
|
===================================================================
|
||||||
|
--- qemu-5.1+dfsg.orig/qapi/block-core.json
|
||||||
|
+++ qemu-5.1+dfsg/qapi/block-core.json
|
||||||
|
@@ -2807,7 +2807,7 @@
|
||||||
|
'luks', 'nbd', 'nfs', 'null-aio', 'null-co', 'nvme', 'parallels',
|
||||||
|
'qcow', 'qcow2', 'qed', 'quorum', 'raw', 'rbd',
|
||||||
|
{ 'name': 'replication', 'if': 'defined(CONFIG_REPLICATION)' },
|
||||||
|
- 'sheepdog',
|
||||||
|
+ 'sheepdog', 'vitastor',
|
||||||
|
'ssh', 'throttle', 'vdi', 'vhdx', 'vmdk', 'vpc', 'vvfat' ] }
|
||||||
|
|
||||||
|
##
|
||||||
|
@@ -3644,6 +3644,24 @@
|
||||||
|
'*tag': 'str' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific block device options for vitastor
|
||||||
|
+#
|
||||||
|
+# @inode: Inode number
|
||||||
|
+# @pool: Pool ID
|
||||||
|
+# @size: Desired image size in bytes
|
||||||
|
+# @etcd_host: etcd connection address
|
||||||
|
+# @etcd_prefix: etcd key/value prefix
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'data': { 'inode': 'uint64',
|
||||||
|
+ 'pool': 'uint64',
|
||||||
|
+ 'size': 'uint64',
|
||||||
|
+ 'etcd_host': 'str',
|
||||||
|
+ '*etcd_prefix': 'str' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @ReplicationMode:
|
||||||
|
#
|
||||||
|
# An enumeration of replication modes.
|
||||||
|
@@ -3988,6 +4006,7 @@
|
||||||
|
'replication': { 'type': 'BlockdevOptionsReplication',
|
||||||
|
'if': 'defined(CONFIG_REPLICATION)' },
|
||||||
|
'sheepdog': 'BlockdevOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevOptionsSsh',
|
||||||
|
'throttle': 'BlockdevOptionsThrottle',
|
||||||
|
'vdi': 'BlockdevOptionsGenericFormat',
|
||||||
|
@@ -4376,6 +4395,17 @@
|
||||||
|
'*cluster-size' : 'size' } }
|
||||||
|
|
||||||
|
##
|
||||||
|
+# @BlockdevCreateOptionsVitastor:
|
||||||
|
+#
|
||||||
|
+# Driver specific image creation options for Vitastor.
|
||||||
|
+#
|
||||||
|
+# @size: Size of the virtual disk in bytes
|
||||||
|
+##
|
||||||
|
+{ 'struct': 'BlockdevCreateOptionsVitastor',
|
||||||
|
+ 'data': { 'location': 'BlockdevOptionsVitastor',
|
||||||
|
+ 'size': 'size' } }
|
||||||
|
+
|
||||||
|
+##
|
||||||
|
# @BlockdevVmdkSubformat:
|
||||||
|
#
|
||||||
|
# Subformat options for VMDK images
|
||||||
|
@@ -4637,6 +4667,7 @@
|
||||||
|
'qed': 'BlockdevCreateOptionsQed',
|
||||||
|
'rbd': 'BlockdevCreateOptionsRbd',
|
||||||
|
'sheepdog': 'BlockdevCreateOptionsSheepdog',
|
||||||
|
+ 'vitastor': 'BlockdevCreateOptionsVitastor',
|
||||||
|
'ssh': 'BlockdevCreateOptionsSsh',
|
||||||
|
'vdi': 'BlockdevCreateOptionsVdi',
|
||||||
|
'vhdx': 'BlockdevCreateOptionsVhdx',
|
||||||
|
Index: qemu-5.1+dfsg/scripts/modules/module_block.py
|
||||||
|
===================================================================
|
||||||
|
--- qemu-5.1+dfsg.orig/scripts/modules/module_block.py
|
||||||
|
+++ qemu-5.1+dfsg/scripts/modules/module_block.py
|
||||||
|
@@ -86,6 +86,7 @@ if __name__ == '__main__':
|
||||||
|
output_file = sys.argv[1]
|
||||||
|
with open(output_file, 'w') as fheader:
|
||||||
|
print_top(fheader)
|
||||||
|
+ add_module(fheader, "vitastor", "vitastor", "vitastor")
|
||||||
|
|
||||||
|
for filename in sys.argv[2:]:
|
||||||
|
if os.path.isfile(filename):
|
|
@ -1,5 +1,5 @@
|
||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
# Vitastor depends on QEMU and/or FIO headers, but QEMU and FIO don't have -devel packages
|
# Vitastor depends on QEMU and FIO headers, but QEMU and FIO don't have -devel packages
|
||||||
# So we have to copy their headers into the source tarball
|
# So we have to copy their headers into the source tarball
|
||||||
|
|
||||||
set -e
|
set -e
|
||||||
|
@ -18,11 +18,34 @@ else
|
||||||
fi
|
fi
|
||||||
cd ~/rpmbuild/SPECS
|
cd ~/rpmbuild/SPECS
|
||||||
rpmbuild -bp fio.spec
|
rpmbuild -bp fio.spec
|
||||||
|
perl -i -pe 's/^make V=1/exit 0; make V=1/' qemu*.spec
|
||||||
|
rpmbuild -bc qemu*.spec
|
||||||
|
perl -i -pe 's/^exit 0; make V=1/make V=1/' qemu*.spec
|
||||||
|
cd ~/rpmbuild/BUILD/qemu*/
|
||||||
|
rm -rf $VITASTOR/qemu $VITASTOR/fio
|
||||||
|
mkdir -p $VITASTOR/qemu/b/qemu
|
||||||
|
make -j8 config-host.h
|
||||||
|
cp config-host.h $VITASTOR/qemu/b/qemu
|
||||||
|
cp -r include $VITASTOR/qemu
|
||||||
|
if [ -f qapi-schema.json ]; then
|
||||||
|
# QEMU 2.0
|
||||||
|
make qapi-types.h
|
||||||
|
cp qapi-types.h $VITASTOR/qemu/b/qemu
|
||||||
|
else
|
||||||
|
# QEMU 3.0+
|
||||||
|
make qapi
|
||||||
|
cp -r qapi $VITASTOR/qemu/b/qemu
|
||||||
|
fi
|
||||||
cd $VITASTOR
|
cd $VITASTOR
|
||||||
|
sh copy-qemu-includes.sh
|
||||||
|
rm -rf qemu
|
||||||
|
mv qemu-copy qemu
|
||||||
ln -s ~/rpmbuild/BUILD/fio*/ fio
|
ln -s ~/rpmbuild/BUILD/fio*/ fio
|
||||||
sh copy-fio-includes.sh
|
sh copy-fio-includes.sh
|
||||||
rm fio
|
rm fio
|
||||||
mv fio-copy fio
|
mv fio-copy fio
|
||||||
FIO=`rpm -qi fio | perl -e 'while(<>) { /^Epoch[\s:]+(\S+)/ && print "$1:"; /^Version[\s:]+(\S+)/ && print $1; /^Release[\s:]+(\S+)/ && print "-$1"; }'`
|
FIO=`rpm -qi fio | perl -e 'while(<>) { /^Epoch[\s:]+(\S+)/ && print "$1:"; /^Version[\s:]+(\S+)/ && print $1; /^Release[\s:]+(\S+)/ && print "-$1"; }'`
|
||||||
|
QEMU=`rpm -qi qemu qemu-kvm | perl -e 'while(<>) { /^Epoch[\s:]+(\S+)/ && print "$1:"; /^Version[\s:]+(\S+)/ && print $1; /^Release[\s:]+(\S+)/ && print "-$1"; }'`
|
||||||
perl -i -pe 's/(Requires:\s*fio)([^\n]+)?/$1 = '$FIO'/' $VITASTOR/rpm/vitastor-el$EL.spec
|
perl -i -pe 's/(Requires:\s*fio)([^\n]+)?/$1 = '$FIO'/' $VITASTOR/rpm/vitastor-el$EL.spec
|
||||||
tar --transform 's#^#vitastor-0.6.10/#' --exclude 'rpm/*.rpm' -czf $VITASTOR/../vitastor-0.6.10$(rpm --eval '%dist').tar.gz *
|
perl -i -pe 's/(Requires:\s*qemu(?:-kvm)?)([^\n]+)?/$1 = '$QEMU'/' $VITASTOR/rpm/vitastor-el$EL.spec
|
||||||
|
tar --transform 's#^#vitastor-0.6.2/#' --exclude 'rpm/*.rpm' -czf $VITASTOR/../vitastor-0.6.2$(rpm --eval '%dist').tar.gz *
|
||||||
|
|
|
@ -1,6 +1,3 @@
|
||||||
# This is an attempt to automatically build patched RPM specs
|
|
||||||
# More or less broken, better use *.spec.patch for now (and copy src/qemu_driver.c to SOURCES/qemu-vitastor.c)
|
|
||||||
|
|
||||||
# Build packages for CentOS 8 inside a container
|
# Build packages for CentOS 8 inside a container
|
||||||
# cd ..; podman build -t qemu-el8 -v `pwd`/packages:/root/packages -f rpm/qemu-el8.Dockerfile .
|
# cd ..; podman build -t qemu-el8 -v `pwd`/packages:/root/packages -f rpm/qemu-el8.Dockerfile .
|
||||||
|
|
||||||
|
@ -14,7 +11,7 @@ RUN rm -rf /var/lib/dnf/*; dnf download --disablerepo='*' --enablerepo='centos-a
|
||||||
RUN rpm --nomd5 -i qemu*.src.rpm
|
RUN rpm --nomd5 -i qemu*.src.rpm
|
||||||
RUN cd ~/rpmbuild/SPECS && dnf builddep -y --enablerepo=PowerTools --spec qemu-kvm.spec
|
RUN cd ~/rpmbuild/SPECS && dnf builddep -y --enablerepo=PowerTools --spec qemu-kvm.spec
|
||||||
|
|
||||||
ADD patches/qemu-*-vitastor.patch /root/vitastor/patches/
|
ADD qemu-*-vitastor.patch /root/vitastor/
|
||||||
|
|
||||||
RUN set -e; \
|
RUN set -e; \
|
||||||
mkdir -p /root/packages/qemu-el8; \
|
mkdir -p /root/packages/qemu-el8; \
|
||||||
|
@ -28,7 +25,7 @@ RUN set -e; \
|
||||||
echo "Patch$((PN+1)): qemu-4.2-vitastor.patch" >> qemu-kvm.spec; \
|
echo "Patch$((PN+1)): qemu-4.2-vitastor.patch" >> qemu-kvm.spec; \
|
||||||
tail -n +2 xx01 >> qemu-kvm.spec; \
|
tail -n +2 xx01 >> qemu-kvm.spec; \
|
||||||
perl -i -pe 's/(^Release:\s*\d+)/$1.vitastor/' qemu-kvm.spec; \
|
perl -i -pe 's/(^Release:\s*\d+)/$1.vitastor/' qemu-kvm.spec; \
|
||||||
cp /root/vitastor/patches/qemu-4.2-vitastor.patch ~/rpmbuild/SOURCES; \
|
cp /root/vitastor/qemu-4.2-vitastor.patch ~/rpmbuild/SOURCES; \
|
||||||
rpmbuild --nocheck -ba qemu-kvm.spec; \
|
rpmbuild --nocheck -ba qemu-kvm.spec; \
|
||||||
cp ~/rpmbuild/RPMS/*/*qemu* /root/packages/qemu-el8/; \
|
cp ~/rpmbuild/RPMS/*/*qemu* /root/packages/qemu-el8/; \
|
||||||
cp ~/rpmbuild/SRPMS/*qemu* /root/packages/qemu-el8/
|
cp ~/rpmbuild/SRPMS/*qemu* /root/packages/qemu-el8/
|
||||||
|
|
|
@ -1,112 +0,0 @@
|
||||||
--- qemu-kvm.spec.orig 2021-12-01 22:10:58.967935539 +0000
|
|
||||||
+++ qemu-kvm.spec 2021-12-01 22:14:38.530117175 +0000
|
|
||||||
@@ -57,6 +57,7 @@ Requires: %{name}-block-gluster = %{epoc
|
|
||||||
%endif \
|
|
||||||
Requires: %{name}-block-iscsi = %{epoch}:%{version}-%{release} \
|
|
||||||
Requires: %{name}-block-rbd = %{epoch}:%{version}-%{release} \
|
|
||||||
+Requires: %{name}-block-vitastor = %{epoch}:%{version}-%{release}\
|
|
||||||
Requires: %{name}-block-ssh = %{epoch}:%{version}-%{release}
|
|
||||||
|
|
||||||
# Macro to properly setup RHEL/RHEV conflict handling
|
|
||||||
@@ -67,7 +68,7 @@ Obsoletes: %1-rhev
|
|
||||||
Summary: QEMU is a machine emulator and virtualizer
|
|
||||||
Name: qemu-kvm
|
|
||||||
Version: 4.2.0
|
|
||||||
-Release: 29%{?dist}.6
|
|
||||||
+Release: 32.vitastor%{?dist}.6
|
|
||||||
# Epoch because we pushed a qemu-1.0 package. AIUI this can't ever be dropped
|
|
||||||
Epoch: 15
|
|
||||||
License: GPLv2 and GPLv2+ and CC-BY
|
|
||||||
@@ -102,6 +103,7 @@ Source33: qemu-pr-helper.socket
|
|
||||||
Source34: 81-kvm-rhel.rules
|
|
||||||
Source35: udev-kvm-check.c
|
|
||||||
Source36: README.tests
|
|
||||||
+Source37: qemu-vitastor.c
|
|
||||||
|
|
||||||
|
|
||||||
Patch0005: 0005-Initial-redhat-build.patch
|
|
||||||
@@ -825,6 +827,7 @@ Patch331: kvm-Drop-bogus-IPv6-messages.p
|
|
||||||
Patch333: kvm-virtiofsd-Whitelist-fchmod.patch
|
|
||||||
# For bz#1883869 - virtiofsd core dump in KATA Container [rhel-8.2.1.z]
|
|
||||||
Patch334: kvm-virtiofsd-avoid-proc-self-fd-tempdir.patch
|
|
||||||
+Patch335: qemu-4.2-vitastor.patch
|
|
||||||
|
|
||||||
BuildRequires: wget
|
|
||||||
BuildRequires: rpm-build
|
|
||||||
@@ -861,6 +864,7 @@ BuildRequires: libcurl-devel
|
|
||||||
BuildRequires: libssh-devel
|
|
||||||
BuildRequires: librados-devel
|
|
||||||
BuildRequires: librbd-devel
|
|
||||||
+BuildRequires: vitastor-client-devel
|
|
||||||
%if %{have_gluster}
|
|
||||||
# For gluster block driver
|
|
||||||
BuildRequires: glusterfs-api-devel
|
|
||||||
@@ -1095,6 +1099,14 @@ Install this package if you want to acce
|
|
||||||
using the rbd protocol.
|
|
||||||
|
|
||||||
|
|
||||||
+%package block-vitastor
|
|
||||||
+Summary: QEMU Vitastor block driver
|
|
||||||
+Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
|
||||||
+
|
|
||||||
+%description block-vitastor
|
|
||||||
+This package provides the additional Vitastor block driver for QEMU.
|
|
||||||
+
|
|
||||||
+
|
|
||||||
%package block-ssh
|
|
||||||
Summary: QEMU SSH block driver
|
|
||||||
Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
|
||||||
@@ -1109,6 +1121,7 @@ the Secure Shell (SSH) protocol.
|
|
||||||
%prep
|
|
||||||
%setup -n qemu-%{version}
|
|
||||||
%autopatch -p1
|
|
||||||
+cp %{SOURCE37} ./block/vitastor.c
|
|
||||||
|
|
||||||
%build
|
|
||||||
%global buildarch %{kvm_target}-softmmu
|
|
||||||
@@ -1116,7 +1129,7 @@ the Secure Shell (SSH) protocol.
|
|
||||||
# --build-id option is used for giving info to the debug packages.
|
|
||||||
buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
|
||||||
|
|
||||||
-%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
|
||||||
+%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,vitastor,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
|
||||||
|
|
||||||
%if 0%{have_gluster}
|
|
||||||
%global block_drivers_list %{block_drivers_list},gluster
|
|
||||||
@@ -1131,7 +1144,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
|
||||||
--docdir="%{qemudocdir}" \
|
|
||||||
--libexecdir="%{_libexecdir}" \
|
|
||||||
--extra-ldflags="-Wl,--build-id -Wl,-z,relro -Wl,-z,now" \
|
|
||||||
- --extra-cflags="%{optflags}" \
|
|
||||||
+ --extra-cflags="%{optflags} -DRHEL_BDRV_CO_TRUNCATE_FLAGS" \
|
|
||||||
--with-pkgversion="%{name}-%{version}-%{release}" \
|
|
||||||
--with-confsuffix=/"%{name}" \
|
|
||||||
--firmwarepath=%{_prefix}/share/qemu-firmware \
|
|
||||||
@@ -1152,6 +1165,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
|
||||||
--disable-numa \
|
|
||||||
%endif
|
|
||||||
--enable-rbd \
|
|
||||||
+ --enable-vitastor \
|
|
||||||
%if 0%{have_librdma}
|
|
||||||
--enable-rdma \
|
|
||||||
%else
|
|
||||||
@@ -1192,9 +1206,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
|
||||||
%endif
|
|
||||||
--python=%{__python3} \
|
|
||||||
--target-list="%{buildarch}" \
|
|
||||||
- --block-drv-rw-whitelist=%{block_drivers_list} \
|
|
||||||
--audio-drv-list= \
|
|
||||||
- --block-drv-ro-whitelist=vmdk,vhdx,vpc,https,ssh \
|
|
||||||
--with-coroutine=ucontext \
|
|
||||||
--tls-priority=NORMAL \
|
|
||||||
--disable-bluez \
|
|
||||||
@@ -1750,6 +1762,9 @@ useradd -r -u 107 -g qemu -G kvm -d / -s
|
|
||||||
%files block-rbd
|
|
||||||
%{_libdir}/qemu-kvm/block-rbd.so
|
|
||||||
|
|
||||||
+%files block-vitastor
|
|
||||||
+%{_libdir}/qemu-kvm/block-vitastor.so
|
|
||||||
+
|
|
||||||
%files block-ssh
|
|
||||||
%{_libdir}/qemu-kvm/block-ssh.so
|
|
||||||
|
|
|
@ -1,103 +0,0 @@
|
||||||
--- qemu-kvm_6.0.spec.orig 2021-10-22 13:22:07.000000000 +0000
|
|
||||||
+++ qemu-kvm_6.0.spec 2021-12-01 22:43:26.095508618 +0000
|
|
||||||
@@ -67,6 +67,7 @@ Requires: %{name}-hw-usbredir = %{epoch}
|
|
||||||
%endif \
|
|
||||||
Requires: %{name}-block-iscsi = %{epoch}:%{version}-%{release} \
|
|
||||||
Requires: %{name}-block-rbd = %{epoch}:%{version}-%{release} \
|
|
||||||
+Requires: %{name}-block-vitastor = %{epoch}:%{version}-%{release}\
|
|
||||||
Requires: %{name}-block-ssh = %{epoch}:%{version}-%{release}
|
|
||||||
|
|
||||||
# Macro to properly setup RHEL/RHEV conflict handling
|
|
||||||
@@ -77,7 +78,7 @@ Obsoletes: %1-rhev <= %{epoch}:%{version
|
|
||||||
Summary: QEMU is a machine emulator and virtualizer
|
|
||||||
Name: qemu-kvm
|
|
||||||
Version: 6.0.0
|
|
||||||
-Release: 33%{?dist}
|
|
||||||
+Release: 33.vitastor%{?dist}
|
|
||||||
# Epoch because we pushed a qemu-1.0 package. AIUI this can't ever be dropped
|
|
||||||
Epoch: 15
|
|
||||||
License: GPLv2 and GPLv2+ and CC-BY
|
|
||||||
@@ -112,6 +113,7 @@ Source33: qemu-pr-helper.socket
|
|
||||||
Source34: 81-kvm-rhel.rules
|
|
||||||
Source35: udev-kvm-check.c
|
|
||||||
Source36: README.tests
|
|
||||||
+Source37: qemu-vitastor.c
|
|
||||||
|
|
||||||
|
|
||||||
Patch0001: 0001-redhat-Adding-slirp-to-the-exploded-tree.patch
|
|
||||||
@@ -342,6 +344,7 @@ Patch109: kvm-virtio-balloon-Fix-page-po
|
|
||||||
Patch110: kvm-virtio-net-fix-use-after-unmap-free-for-sg.patch
|
|
||||||
# For bz#1999141 - migration fails with: "qemu-kvm: get_pci_config_device: Bad config data: i=0x9a read: 3 device: 2 cmask: ff wmask: 0 w1cmask:0"
|
|
||||||
Patch111: kvm-Fix-virtio-net-pci-vectors-compat.patch
|
|
||||||
+Patch112: qemu-6.0-vitastor.patch
|
|
||||||
|
|
||||||
BuildRequires: wget
|
|
||||||
BuildRequires: rpm-build
|
|
||||||
@@ -379,6 +382,7 @@ BuildRequires: libcurl-devel
|
|
||||||
BuildRequires: libssh-devel
|
|
||||||
BuildRequires: librados-devel
|
|
||||||
BuildRequires: librbd-devel
|
|
||||||
+BuildRequires: vitastor-client-devel
|
|
||||||
%if %{have_gluster}
|
|
||||||
# For gluster block driver
|
|
||||||
BuildRequires: glusterfs-api-devel
|
|
||||||
@@ -625,6 +629,14 @@ Install this package if you want to acce
|
|
||||||
using the rbd protocol.
|
|
||||||
|
|
||||||
|
|
||||||
+%package block-vitastor
|
|
||||||
+Summary: QEMU Vitastor block driver
|
|
||||||
+Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
|
||||||
+
|
|
||||||
+%description block-vitastor
|
|
||||||
+This package provides the additional Vitastor block driver for QEMU.
|
|
||||||
+
|
|
||||||
+
|
|
||||||
%package block-ssh
|
|
||||||
Summary: QEMU SSH block driver
|
|
||||||
Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
|
||||||
@@ -678,6 +690,7 @@ This package provides usbredir support.
|
|
||||||
rm -fr slirp
|
|
||||||
mkdir slirp
|
|
||||||
%autopatch -p1
|
|
||||||
+cp %{SOURCE37} ./block/vitastor.c
|
|
||||||
|
|
||||||
%global qemu_kvm_build qemu_kvm_build
|
|
||||||
%global qemu_kiwi_build qemu_kiwi_src/build
|
|
||||||
@@ -701,7 +714,7 @@ mkdir -p %{qemu_kvm_build}
|
|
||||||
# --build-id option is used for giving info to the debug packages.
|
|
||||||
buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
|
||||||
|
|
||||||
-%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
|
||||||
+%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,vitastor,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
|
||||||
|
|
||||||
%if 0%{have_gluster}
|
|
||||||
%global block_drivers_list %{block_drivers_list},gluster
|
|
||||||
@@ -894,6 +907,7 @@ pushd %{qemu_kvm_build}
|
|
||||||
%endif
|
|
||||||
--enable-pie \
|
|
||||||
--enable-rbd \
|
|
||||||
+ --enable-vitastor \
|
|
||||||
%if 0%{have_librdma}
|
|
||||||
--enable-rdma \
|
|
||||||
%endif
|
|
||||||
@@ -977,9 +991,7 @@ find ../default-configs -name "*-rh-devi
|
|
||||||
--firmwarepath=%{_prefix}/share/qemu-firmware \
|
|
||||||
--meson="%{__meson}" \
|
|
||||||
--target-list="%{buildarch}" \
|
|
||||||
- --block-drv-rw-whitelist=%{block_drivers_list} \
|
|
||||||
--audio-drv-list= \
|
|
||||||
- --block-drv-ro-whitelist=vmdk,vhdx,vpc,https,ssh \
|
|
||||||
--with-coroutine=ucontext \
|
|
||||||
--with-git=git \
|
|
||||||
--tls-priority=@QEMU,SYSTEM \
|
|
||||||
@@ -1584,6 +1596,9 @@ sh %{_sysconfdir}/sysconfig/modules/kvm.
|
|
||||||
%files block-rbd
|
|
||||||
%{_libdir}/qemu-kvm/block-rbd.so
|
|
||||||
|
|
||||||
+%files block-vitastor
|
|
||||||
+%{_libdir}/qemu-kvm/block-vitastor.so
|
|
||||||
+
|
|
||||||
%files block-ssh
|
|
||||||
%{_libdir}/qemu-kvm/block-ssh.so
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
--- qemu-kvm.spec.orig 2020-11-09 23:41:03.000000000 +0000
|
--- qemu-kvm.spec.orig 2020-11-09 23:41:03.000000000 +0000
|
||||||
+++ qemu-kvm.spec 2021-12-01 21:53:30.895747529 +0000
|
+++ qemu-kvm.spec 2020-12-06 10:44:24.207640963 +0000
|
||||||
@@ -2,7 +2,7 @@
|
@@ -2,7 +2,7 @@
|
||||||
%global SLOF_gittagcommit 899d9883
|
%global SLOF_gittagcommit 899d9883
|
||||||
|
|
||||||
|
@ -9,24 +9,25 @@
|
||||||
%global have_opengl 1
|
%global have_opengl 1
|
||||||
%global have_fdt 0
|
%global have_fdt 0
|
||||||
%global have_gluster 1
|
%global have_gluster 1
|
||||||
@@ -57,6 +57,7 @@ Requires: %{name}-block-gluster = %{epoc
|
@@ -56,7 +56,7 @@ Requires: %{name}-block-curl = %{epoch}:
|
||||||
|
Requires: %{name}-block-gluster = %{epoch}:%{version}-%{release} \
|
||||||
%endif \
|
%endif \
|
||||||
Requires: %{name}-block-iscsi = %{epoch}:%{version}-%{release} \
|
Requires: %{name}-block-iscsi = %{epoch}:%{version}-%{release} \
|
||||||
Requires: %{name}-block-rbd = %{epoch}:%{version}-%{release} \
|
-Requires: %{name}-block-rbd = %{epoch}:%{version}-%{release} \
|
||||||
+Requires: %{name}-block-vitastor = %{epoch}:%{version}-%{release}\
|
+#Requires: %{name}-block-rbd = %{epoch}:%{version}-%{release} \
|
||||||
Requires: %{name}-block-ssh = %{epoch}:%{version}-%{release}
|
Requires: %{name}-block-ssh = %{epoch}:%{version}-%{release}
|
||||||
|
|
||||||
# Macro to properly setup RHEL/RHEV conflict handling
|
# Macro to properly setup RHEL/RHEV conflict handling
|
||||||
@@ -67,7 +68,7 @@ Obsoletes: %1-rhev
|
@@ -67,7 +67,7 @@ Obsoletes: %1-rhev
|
||||||
Summary: QEMU is a machine emulator and virtualizer
|
Summary: QEMU is a machine emulator and virtualizer
|
||||||
Name: qemu-kvm
|
Name: qemu-kvm
|
||||||
Version: 4.2.0
|
Version: 4.2.0
|
||||||
-Release: 29.vitastor%{?dist}.6
|
-Release: 29.vitastor%{?dist}.6
|
||||||
+Release: 32.vitastor%{?dist}.6
|
+Release: 30.vitastor%{?dist}.6
|
||||||
# Epoch because we pushed a qemu-1.0 package. AIUI this can't ever be dropped
|
# Epoch because we pushed a qemu-1.0 package. AIUI this can't ever be dropped
|
||||||
Epoch: 15
|
Epoch: 15
|
||||||
License: GPLv2 and GPLv2+ and CC-BY
|
License: GPLv2 and GPLv2+ and CC-BY
|
||||||
@@ -99,9 +100,10 @@ Source30: kvm-s390x.conf
|
@@ -99,8 +99,8 @@ Source30: kvm-s390x.conf
|
||||||
Source31: kvm-x86.conf
|
Source31: kvm-x86.conf
|
||||||
Source32: qemu-pr-helper.service
|
Source32: qemu-pr-helper.service
|
||||||
Source33: qemu-pr-helper.socket
|
Source33: qemu-pr-helper.socket
|
||||||
|
@ -35,11 +36,9 @@
|
||||||
+#Source34: 81-kvm-rhel.rules
|
+#Source34: 81-kvm-rhel.rules
|
||||||
+#Source35: udev-kvm-check.c
|
+#Source35: udev-kvm-check.c
|
||||||
Source36: README.tests
|
Source36: README.tests
|
||||||
+Source37: qemu-vitastor.c
|
|
||||||
|
|
||||||
|
|
||||||
Patch0005: 0005-Initial-redhat-build.patch
|
@@ -825,7 +825,9 @@ Patch331: kvm-Drop-bogus-IPv6-messages.p
|
||||||
@@ -825,7 +827,9 @@ Patch331: kvm-Drop-bogus-IPv6-messages.p
|
|
||||||
Patch333: kvm-virtiofsd-Whitelist-fchmod.patch
|
Patch333: kvm-virtiofsd-Whitelist-fchmod.patch
|
||||||
# For bz#1883869 - virtiofsd core dump in KATA Container [rhel-8.2.1.z]
|
# For bz#1883869 - virtiofsd core dump in KATA Container [rhel-8.2.1.z]
|
||||||
Patch334: kvm-virtiofsd-avoid-proc-self-fd-tempdir.patch
|
Patch334: kvm-virtiofsd-avoid-proc-self-fd-tempdir.patch
|
||||||
|
@ -50,7 +49,7 @@
|
||||||
|
|
||||||
BuildRequires: wget
|
BuildRequires: wget
|
||||||
BuildRequires: rpm-build
|
BuildRequires: rpm-build
|
||||||
@@ -842,7 +846,8 @@ BuildRequires: pciutils-devel
|
@@ -842,7 +844,8 @@ BuildRequires: pciutils-devel
|
||||||
BuildRequires: libiscsi-devel
|
BuildRequires: libiscsi-devel
|
||||||
BuildRequires: ncurses-devel
|
BuildRequires: ncurses-devel
|
||||||
BuildRequires: libattr-devel
|
BuildRequires: libattr-devel
|
||||||
|
@ -60,7 +59,7 @@
|
||||||
%if %{have_usbredir}
|
%if %{have_usbredir}
|
||||||
BuildRequires: usbredir-devel >= 0.7.1
|
BuildRequires: usbredir-devel >= 0.7.1
|
||||||
%endif
|
%endif
|
||||||
@@ -856,12 +861,13 @@ BuildRequires: virglrenderer-devel
|
@@ -856,12 +859,12 @@ BuildRequires: virglrenderer-devel
|
||||||
# For smartcard NSS support
|
# For smartcard NSS support
|
||||||
BuildRequires: nss-devel
|
BuildRequires: nss-devel
|
||||||
%endif
|
%endif
|
||||||
|
@ -71,13 +70,12 @@
|
||||||
BuildRequires: libssh-devel
|
BuildRequires: libssh-devel
|
||||||
-BuildRequires: librados-devel
|
-BuildRequires: librados-devel
|
||||||
-BuildRequires: librbd-devel
|
-BuildRequires: librbd-devel
|
||||||
+BuildRequires: librados2-devel
|
+#BuildRequires: librados-devel
|
||||||
+BuildRequires: librbd1-devel
|
+#BuildRequires: librbd-devel
|
||||||
+BuildRequires: vitastor-client-devel
|
|
||||||
%if %{have_gluster}
|
%if %{have_gluster}
|
||||||
# For gluster block driver
|
# For gluster block driver
|
||||||
BuildRequires: glusterfs-api-devel
|
BuildRequires: glusterfs-api-devel
|
||||||
@@ -955,25 +961,25 @@ hardware for a full system such as a PC
|
@@ -955,25 +958,25 @@ hardware for a full system such as a PC
|
||||||
|
|
||||||
%package -n qemu-kvm-core
|
%package -n qemu-kvm-core
|
||||||
Summary: qemu-kvm core components
|
Summary: qemu-kvm core components
|
||||||
|
@ -107,35 +105,38 @@
|
||||||
# For compressed guest memory dumps
|
# For compressed guest memory dumps
|
||||||
Requires: lzo snappy
|
Requires: lzo snappy
|
||||||
%if %{have_kvm_setup}
|
%if %{have_kvm_setup}
|
||||||
@@ -1096,6 +1102,14 @@ Install this package if you want to acce
|
@@ -1085,15 +1088,15 @@ This package provides the additional iSC
|
||||||
using the rbd protocol.
|
Install this package if you want to access iSCSI volumes.
|
||||||
|
|
||||||
|
|
||||||
|
-%package block-rbd
|
||||||
|
-Summary: QEMU Ceph/RBD block driver
|
||||||
|
-Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
||||||
|
-
|
||||||
|
-%description block-rbd
|
||||||
|
-This package provides the additional Ceph/RBD block driver for QEMU.
|
||||||
|
-
|
||||||
|
-Install this package if you want to access remote Ceph volumes
|
||||||
|
-using the rbd protocol.
|
||||||
|
+#%package block-rbd
|
||||||
|
+#Summary: QEMU Ceph/RBD block driver
|
||||||
|
+#Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
||||||
|
+#
|
||||||
|
+#%description block-rbd
|
||||||
|
+#This package provides the additional Ceph/RBD block driver for QEMU.
|
||||||
|
+#
|
||||||
|
+#Install this package if you want to access remote Ceph volumes
|
||||||
|
+#using the rbd protocol.
|
||||||
|
|
||||||
|
|
||||||
+%package block-vitastor
|
|
||||||
+Summary: QEMU Vitastor block driver
|
|
||||||
+Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
|
||||||
+
|
|
||||||
+%description block-vitastor
|
|
||||||
+This package provides the additional Vitastor block driver for QEMU.
|
|
||||||
+
|
|
||||||
+
|
|
||||||
%package block-ssh
|
%package block-ssh
|
||||||
Summary: QEMU SSH block driver
|
@@ -1117,12 +1120,14 @@ the Secure Shell (SSH) protocol.
|
||||||
Requires: %{name}-common%{?_isa} = %{epoch}:%{version}-%{release}
|
|
||||||
@@ -1110,6 +1124,7 @@ the Secure Shell (SSH) protocol.
|
|
||||||
%prep
|
|
||||||
%setup -n qemu-%{version}
|
|
||||||
%autopatch -p1
|
|
||||||
+cp %{SOURCE37} ./block/vitastor.c
|
|
||||||
|
|
||||||
%build
|
|
||||||
%global buildarch %{kvm_target}-softmmu
|
|
||||||
@@ -1117,12 +1132,13 @@ the Secure Shell (SSH) protocol.
|
|
||||||
# --build-id option is used for giving info to the debug packages.
|
# --build-id option is used for giving info to the debug packages.
|
||||||
buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
||||||
|
|
||||||
-%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
-%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
||||||
+%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,vitastor,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
+#%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,rbd,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
||||||
|
+%global block_drivers_list qcow2,raw,file,host_device,nbd,iscsi,blkdebug,luks,null-co,nvme,copy-on-read,throttle
|
||||||
|
|
||||||
%if 0%{have_gluster}
|
%if 0%{have_gluster}
|
||||||
%global block_drivers_list %{block_drivers_list},gluster
|
%global block_drivers_list %{block_drivers_list},gluster
|
||||||
|
@ -145,20 +146,12 @@
|
||||||
./configure \
|
./configure \
|
||||||
--prefix="%{_prefix}" \
|
--prefix="%{_prefix}" \
|
||||||
--libdir="%{_libdir}" \
|
--libdir="%{_libdir}" \
|
||||||
@@ -1132,7 +1148,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
@@ -1152,15 +1157,15 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
||||||
--docdir="%{qemudocdir}" \
|
%else
|
||||||
--libexecdir="%{_libexecdir}" \
|
|
||||||
--extra-ldflags="-Wl,--build-id -Wl,-z,relro -Wl,-z,now" \
|
|
||||||
- --extra-cflags="%{optflags}" \
|
|
||||||
+ --extra-cflags="%{optflags} -DRHEL_BDRV_CO_TRUNCATE_FLAGS" \
|
|
||||||
--with-pkgversion="%{name}-%{version}-%{release}" \
|
|
||||||
--with-confsuffix=/"%{name}" \
|
|
||||||
--firmwarepath=%{_prefix}/share/qemu-firmware \
|
|
||||||
@@ -1153,14 +1169,15 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
|
||||||
--disable-numa \
|
--disable-numa \
|
||||||
%endif
|
%endif
|
||||||
--enable-rbd \
|
- --enable-rbd \
|
||||||
+ --enable-vitastor \
|
+ --disable-rbd \
|
||||||
%if 0%{have_librdma}
|
%if 0%{have_librdma}
|
||||||
--enable-rdma \
|
--enable-rdma \
|
||||||
%else
|
%else
|
||||||
|
@ -172,7 +165,7 @@
|
||||||
--enable-spice \
|
--enable-spice \
|
||||||
--enable-smartcard \
|
--enable-smartcard \
|
||||||
--enable-virglrenderer \
|
--enable-virglrenderer \
|
||||||
@@ -1179,7 +1196,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
@@ -1179,7 +1184,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
||||||
%else
|
%else
|
||||||
--disable-usb-redir \
|
--disable-usb-redir \
|
||||||
%endif
|
%endif
|
||||||
|
@ -181,7 +174,7 @@
|
||||||
%ifarch x86_64
|
%ifarch x86_64
|
||||||
--enable-libpmem \
|
--enable-libpmem \
|
||||||
%else
|
%else
|
||||||
@@ -1193,9 +1210,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
@@ -1193,9 +1198,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
||||||
%endif
|
%endif
|
||||||
--python=%{__python3} \
|
--python=%{__python3} \
|
||||||
--target-list="%{buildarch}" \
|
--target-list="%{buildarch}" \
|
||||||
|
@ -191,7 +184,7 @@
|
||||||
--with-coroutine=ucontext \
|
--with-coroutine=ucontext \
|
||||||
--tls-priority=NORMAL \
|
--tls-priority=NORMAL \
|
||||||
--disable-bluez \
|
--disable-bluez \
|
||||||
@@ -1262,7 +1277,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
@@ -1262,7 +1265,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
||||||
--disable-sanitizers \
|
--disable-sanitizers \
|
||||||
--disable-hvf \
|
--disable-hvf \
|
||||||
--disable-whpx \
|
--disable-whpx \
|
||||||
|
@ -200,7 +193,7 @@
|
||||||
--disable-membarrier \
|
--disable-membarrier \
|
||||||
--disable-vhost-crypto \
|
--disable-vhost-crypto \
|
||||||
--disable-libxml2 \
|
--disable-libxml2 \
|
||||||
@@ -1308,7 +1323,7 @@ make V=1 %{?_smp_mflags} $buildldflags
|
@@ -1308,7 +1311,7 @@ make V=1 %{?_smp_mflags} $buildldflags
|
||||||
cp -a %{kvm_target}-softmmu/qemu-system-%{kvm_target} qemu-kvm
|
cp -a %{kvm_target}-softmmu/qemu-system-%{kvm_target} qemu-kvm
|
||||||
|
|
||||||
gcc %{SOURCE6} $RPM_OPT_FLAGS $RPM_LD_FLAGS -o ksmctl
|
gcc %{SOURCE6} $RPM_OPT_FLAGS $RPM_LD_FLAGS -o ksmctl
|
||||||
|
@ -209,7 +202,7 @@
|
||||||
|
|
||||||
%install
|
%install
|
||||||
%define _udevdir %(pkg-config --variable=udevdir udev)
|
%define _udevdir %(pkg-config --variable=udevdir udev)
|
||||||
@@ -1343,8 +1358,8 @@ mkdir -p $RPM_BUILD_ROOT%{testsdir}/test
|
@@ -1343,8 +1346,8 @@ mkdir -p $RPM_BUILD_ROOT%{testsdir}/test
|
||||||
mkdir -p $RPM_BUILD_ROOT%{testsdir}/tests/qemu-iotests
|
mkdir -p $RPM_BUILD_ROOT%{testsdir}/tests/qemu-iotests
|
||||||
mkdir -p $RPM_BUILD_ROOT%{testsdir}/scripts/qmp
|
mkdir -p $RPM_BUILD_ROOT%{testsdir}/scripts/qmp
|
||||||
|
|
||||||
|
@ -220,7 +213,7 @@
|
||||||
|
|
||||||
install -m 0644 scripts/dump-guest-memory.py \
|
install -m 0644 scripts/dump-guest-memory.py \
|
||||||
$RPM_BUILD_ROOT%{_datadir}/%{name}
|
$RPM_BUILD_ROOT%{_datadir}/%{name}
|
||||||
@@ -1562,6 +1577,8 @@ rm -rf $RPM_BUILD_ROOT%{qemudocdir}/inte
|
@@ -1562,6 +1565,8 @@ rm -rf $RPM_BUILD_ROOT%{qemudocdir}/inte
|
||||||
# Remove spec
|
# Remove spec
|
||||||
rm -rf $RPM_BUILD_ROOT%{qemudocdir}/specs
|
rm -rf $RPM_BUILD_ROOT%{qemudocdir}/specs
|
||||||
|
|
||||||
|
@ -229,7 +222,7 @@
|
||||||
%check
|
%check
|
||||||
export DIFF=diff; make check V=1
|
export DIFF=diff; make check V=1
|
||||||
|
|
||||||
@@ -1645,8 +1662,8 @@ useradd -r -u 107 -g qemu -G kvm -d / -s
|
@@ -1645,8 +1650,8 @@ useradd -r -u 107 -g qemu -G kvm -d / -s
|
||||||
%config(noreplace) %{_sysconfdir}/sysconfig/ksm
|
%config(noreplace) %{_sysconfdir}/sysconfig/ksm
|
||||||
%{_unitdir}/ksmtuned.service
|
%{_unitdir}/ksmtuned.service
|
||||||
%{_sbindir}/ksmtuned
|
%{_sbindir}/ksmtuned
|
||||||
|
@ -240,7 +233,7 @@
|
||||||
%ghost %{_sysconfdir}/kvm
|
%ghost %{_sysconfdir}/kvm
|
||||||
%config(noreplace) %{_sysconfdir}/ksmtuned.conf
|
%config(noreplace) %{_sysconfdir}/ksmtuned.conf
|
||||||
%dir %{_sysconfdir}/%{name}
|
%dir %{_sysconfdir}/%{name}
|
||||||
@@ -1711,8 +1728,8 @@ useradd -r -u 107 -g qemu -G kvm -d / -s
|
@@ -1711,8 +1716,8 @@ useradd -r -u 107 -g qemu -G kvm -d / -s
|
||||||
%{_libexecdir}/vhost-user-gpu
|
%{_libexecdir}/vhost-user-gpu
|
||||||
%{_datadir}/%{name}/vhost-user/50-qemu-gpu.json
|
%{_datadir}/%{name}/vhost-user/50-qemu-gpu.json
|
||||||
%endif
|
%endif
|
||||||
|
@ -251,13 +244,14 @@
|
||||||
|
|
||||||
%files -n qemu-img
|
%files -n qemu-img
|
||||||
%defattr(-,root,root)
|
%defattr(-,root,root)
|
||||||
@@ -1751,6 +1768,9 @@ useradd -r -u 107 -g qemu -G kvm -d / -s
|
@@ -1748,8 +1753,8 @@ useradd -r -u 107 -g qemu -G kvm -d / -s
|
||||||
%files block-rbd
|
%files block-iscsi
|
||||||
%{_libdir}/qemu-kvm/block-rbd.so
|
%{_libdir}/qemu-kvm/block-iscsi.so
|
||||||
|
|
||||||
|
-%files block-rbd
|
||||||
|
-%{_libdir}/qemu-kvm/block-rbd.so
|
||||||
|
+#%files block-rbd
|
||||||
|
+#%{_libdir}/qemu-kvm/block-rbd.so
|
||||||
|
|
||||||
+%files block-vitastor
|
|
||||||
+%{_libdir}/qemu-kvm/block-vitastor.so
|
|
||||||
+
|
|
||||||
%files block-ssh
|
%files block-ssh
|
||||||
%{_libdir}/qemu-kvm/block-ssh.so
|
%{_libdir}/qemu-kvm/block-ssh.so
|
||||||
|
|
|
@ -0,0 +1,29 @@
|
||||||
|
--- qemu-kvm.spec 2020-12-05 13:13:54.388623517 +0000
|
||||||
|
+++ qemu-kvm.spec 2020-12-05 13:13:58.728696598 +0000
|
||||||
|
@@ -67,7 +67,7 @@ Obsoletes: %1-rhev
|
||||||
|
Summary: QEMU is a machine emulator and virtualizer
|
||||||
|
Name: qemu-kvm
|
||||||
|
Version: 4.2.0
|
||||||
|
-Release: 29%{?dist}.6
|
||||||
|
+Release: 29.vitastor%{?dist}.6
|
||||||
|
# Epoch because we pushed a qemu-1.0 package. AIUI this can't ever be dropped
|
||||||
|
Epoch: 15
|
||||||
|
License: GPLv2 and GPLv2+ and CC-BY
|
||||||
|
@@ -825,6 +825,7 @@ Patch331: kvm-Drop-bogus-IPv6-messages.p
|
||||||
|
Patch333: kvm-virtiofsd-Whitelist-fchmod.patch
|
||||||
|
# For bz#1883869 - virtiofsd core dump in KATA Container [rhel-8.2.1.z]
|
||||||
|
Patch334: kvm-virtiofsd-avoid-proc-self-fd-tempdir.patch
|
||||||
|
+Patch335: qemu-4.2-vitastor.patch
|
||||||
|
|
||||||
|
BuildRequires: wget
|
||||||
|
BuildRequires: rpm-build
|
||||||
|
@@ -1192,9 +1193,7 @@ buildldflags="VL_LDFLAGS=-Wl,--build-id"
|
||||||
|
%endif
|
||||||
|
--python=%{__python3} \
|
||||||
|
--target-list="%{buildarch}" \
|
||||||
|
- --block-drv-rw-whitelist=%{block_drivers_list} \
|
||||||
|
--audio-drv-list= \
|
||||||
|
- --block-drv-ro-whitelist=vmdk,vhdx,vpc,https,ssh \
|
||||||
|
--with-coroutine=ucontext \
|
||||||
|
--tls-priority=NORMAL \
|
||||||
|
--disable-bluez \
|
|
@ -9,11 +9,14 @@ WORKDIR /root
|
||||||
RUN rm -f /etc/yum.repos.d/CentOS-Media.repo
|
RUN rm -f /etc/yum.repos.d/CentOS-Media.repo
|
||||||
RUN yum -y --enablerepo=extras install centos-release-scl epel-release yum-utils rpm-build
|
RUN yum -y --enablerepo=extras install centos-release-scl epel-release yum-utils rpm-build
|
||||||
RUN yum -y install https://vitastor.io/rpms/centos/7/vitastor-release-1.0-1.el7.noarch.rpm
|
RUN yum -y install https://vitastor.io/rpms/centos/7/vitastor-release-1.0-1.el7.noarch.rpm
|
||||||
RUN yum -y install devtoolset-9-gcc-c++ devtoolset-9-libatomic-devel gcc make cmake gperftools-devel fio rh-nodejs12 jerasure-devel gf-complete-devel rdma-core-devel
|
RUN yum -y install devtoolset-9-gcc-c++ devtoolset-9-libatomic-devel gperftools-devel qemu-kvm fio rh-nodejs12 jerasure-devel gf-complete-devel
|
||||||
|
RUN yumdownloader --disablerepo=centos-sclo-rh --source qemu-kvm
|
||||||
RUN yumdownloader --disablerepo=centos-sclo-rh --source fio
|
RUN yumdownloader --disablerepo=centos-sclo-rh --source fio
|
||||||
|
RUN rpm --nomd5 -i qemu*.src.rpm
|
||||||
RUN rpm --nomd5 -i fio*.src.rpm
|
RUN rpm --nomd5 -i fio*.src.rpm
|
||||||
RUN rm -f /etc/yum.repos.d/CentOS-Media.repo
|
RUN rm -f /etc/yum.repos.d/CentOS-Media.repo
|
||||||
RUN cd ~/rpmbuild/SPECS && yum-builddep -y fio.spec
|
RUN cd ~/rpmbuild/SPECS && yum-builddep -y --enablerepo='*' --disablerepo=centos-sclo-rh --disablerepo=centos-sclo-rh-source --disablerepo=centos-sclo-sclo-testing qemu-kvm.spec
|
||||||
|
RUN cd ~/rpmbuild/SPECS && yum-builddep -y --enablerepo='*' --disablerepo=centos-sclo-rh --disablerepo=centos-sclo-rh-source --disablerepo=centos-sclo-sclo-testing fio.spec
|
||||||
|
|
||||||
ADD https://vitastor.io/rpms/liburing-el7/liburing-0.7-2.el7.src.rpm /root
|
ADD https://vitastor.io/rpms/liburing-el7/liburing-0.7-2.el7.src.rpm /root
|
||||||
|
|
||||||
|
@ -34,7 +37,7 @@ ADD . /root/vitastor
|
||||||
RUN set -e; \
|
RUN set -e; \
|
||||||
cd /root/vitastor/rpm; \
|
cd /root/vitastor/rpm; \
|
||||||
sh build-tarball.sh; \
|
sh build-tarball.sh; \
|
||||||
cp /root/vitastor-0.6.10.el7.tar.gz ~/rpmbuild/SOURCES; \
|
cp /root/vitastor-0.6.2.el7.tar.gz ~/rpmbuild/SOURCES; \
|
||||||
cp vitastor-el7.spec ~/rpmbuild/SPECS/vitastor.spec; \
|
cp vitastor-el7.spec ~/rpmbuild/SPECS/vitastor.spec; \
|
||||||
cd ~/rpmbuild/SPECS/; \
|
cd ~/rpmbuild/SPECS/; \
|
||||||
rpmbuild -ba vitastor.spec; \
|
rpmbuild -ba vitastor.spec; \
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
Name: vitastor
|
Name: vitastor
|
||||||
Version: 0.6.10
|
Version: 0.6.2
|
||||||
Release: 1%{?dist}
|
Release: 1%{?dist}
|
||||||
Summary: Vitastor, a fast software-defined clustered block storage
|
Summary: Vitastor, a fast software-defined clustered block storage
|
||||||
|
|
||||||
License: Vitastor Network Public License 1.1
|
License: Vitastor Network Public License 1.1
|
||||||
URL: https://vitastor.io/
|
URL: https://vitastor.io/
|
||||||
Source0: vitastor-0.6.10.el7.tar.gz
|
Source0: vitastor-0.6.2.el7.tar.gz
|
||||||
|
|
||||||
BuildRequires: liburing-devel >= 0.6
|
BuildRequires: liburing-devel >= 0.6
|
||||||
BuildRequires: gperftools-devel
|
BuildRequires: gperftools-devel
|
||||||
|
@ -14,13 +14,14 @@ BuildRequires: rh-nodejs12
|
||||||
BuildRequires: rh-nodejs12-npm
|
BuildRequires: rh-nodejs12-npm
|
||||||
BuildRequires: jerasure-devel
|
BuildRequires: jerasure-devel
|
||||||
BuildRequires: gf-complete-devel
|
BuildRequires: gf-complete-devel
|
||||||
BuildRequires: libibverbs-devel
|
|
||||||
BuildRequires: cmake
|
BuildRequires: cmake
|
||||||
Requires: vitastor-osd = %{version}-%{release}
|
Requires: fio = 3.7-1.el7
|
||||||
Requires: vitastor-mon = %{version}-%{release}
|
Requires: qemu-kvm = 2.0.0-1.el7.6
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
Requires: rh-nodejs12
|
||||||
Requires: vitastor-client-devel = %{version}-%{release}
|
Requires: rh-nodejs12-npm
|
||||||
Requires: vitastor-fio = %{version}-%{release}
|
Requires: liburing >= 0.6
|
||||||
|
Requires: libJerasure2
|
||||||
|
Requires: lpsolve
|
||||||
|
|
||||||
%description
|
%description
|
||||||
Vitastor is a small, simple and fast clustered block storage (storage for VM drives),
|
Vitastor is a small, simple and fast clustered block storage (storage for VM drives),
|
||||||
|
@ -29,66 +30,13 @@ symmetric clustering and automatic data distribution over any number of drives o
|
||||||
size with configurable redundancy (replication or erasure codes/XOR).
|
size with configurable redundancy (replication or erasure codes/XOR).
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-osd
|
|
||||||
Summary: Vitastor - OSD
|
|
||||||
Requires: libJerasure2
|
|
||||||
Requires: liburing >= 0.6
|
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-osd
|
|
||||||
Vitastor object storage daemon, i.e. server program that stores data.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-mon
|
|
||||||
Summary: Vitastor - monitor
|
|
||||||
Requires: rh-nodejs12
|
|
||||||
Requires: rh-nodejs12-npm
|
|
||||||
Requires: lpsolve
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-mon
|
|
||||||
Vitastor monitor, i.e. server program responsible for watching cluster state and
|
|
||||||
scheduling cluster-level operations.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-client
|
|
||||||
Summary: Vitastor - client
|
|
||||||
Requires: liburing >= 0.6
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-client
|
|
||||||
Vitastor client library and command-line interface.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-client-devel
|
|
||||||
Summary: Vitastor - development files
|
|
||||||
Group: Development/Libraries
|
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-client-devel
|
|
||||||
Vitastor library headers for development.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-fio
|
|
||||||
Summary: Vitastor - fio drivers
|
|
||||||
Group: Development/Libraries
|
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
|
||||||
Requires: fio = 3.7-1.el7
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-fio
|
|
||||||
Vitastor fio drivers for benchmarking.
|
|
||||||
|
|
||||||
|
|
||||||
%prep
|
%prep
|
||||||
%setup -q
|
%setup -q
|
||||||
|
|
||||||
|
|
||||||
%build
|
%build
|
||||||
. /opt/rh/devtoolset-9/enable
|
. /opt/rh/devtoolset-9/enable
|
||||||
%cmake .
|
%cmake . -DQEMU_PLUGINDIR=qemu-kvm
|
||||||
%make_build
|
%make_build
|
||||||
|
|
||||||
|
|
||||||
|
@ -100,42 +48,22 @@ cd mon
|
||||||
npm install
|
npm install
|
||||||
cd ..
|
cd ..
|
||||||
mkdir -p %buildroot/usr/lib/vitastor
|
mkdir -p %buildroot/usr/lib/vitastor
|
||||||
cp mon/make-osd.sh %buildroot/usr/lib/vitastor
|
cp -r mon %buildroot/usr/lib/vitastor/mon
|
||||||
cp -r mon %buildroot/usr/lib/vitastor
|
|
||||||
|
|
||||||
|
|
||||||
%files
|
%files
|
||||||
%doc GPL-2.0.txt VNPL-1.1.txt README.md README-ru.md
|
%doc
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-osd
|
|
||||||
%_bindir/vitastor-osd
|
|
||||||
%_bindir/vitastor-dump-journal
|
%_bindir/vitastor-dump-journal
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-mon
|
|
||||||
/usr/lib/vitastor/mon
|
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-client
|
|
||||||
%_bindir/vitastor-nbd
|
%_bindir/vitastor-nbd
|
||||||
%_bindir/vitastor-cli
|
%_bindir/vitastor-osd
|
||||||
%_bindir/vitastor-rm
|
%_bindir/vitastor-rm
|
||||||
%_bindir/vita
|
%_libdir/qemu-kvm/block-vitastor.so
|
||||||
%_libdir/libvitastor_blk.so*
|
|
||||||
%_libdir/libvitastor_client.so*
|
|
||||||
/usr/lib/vitastor/make-osd.sh
|
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-client-devel
|
|
||||||
%_includedir/vitastor_c.h
|
|
||||||
%_libdir/pkgconfig
|
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-fio
|
|
||||||
%_libdir/libfio_vitastor.so
|
%_libdir/libfio_vitastor.so
|
||||||
%_libdir/libfio_vitastor_blk.so
|
%_libdir/libfio_vitastor_blk.so
|
||||||
%_libdir/libfio_vitastor_sec.so
|
%_libdir/libfio_vitastor_sec.so
|
||||||
|
%_libdir/libvitastor_blk.so
|
||||||
|
%_libdir/libvitastor_client.so
|
||||||
|
/usr/lib/vitastor
|
||||||
|
|
||||||
|
|
||||||
%changelog
|
%changelog
|
||||||
|
|
|
@ -8,11 +8,13 @@ WORKDIR /root
|
||||||
RUN rm -f /etc/yum.repos.d/CentOS-Media.repo
|
RUN rm -f /etc/yum.repos.d/CentOS-Media.repo
|
||||||
RUN dnf -y install centos-release-advanced-virtualization epel-release dnf-plugins-core
|
RUN dnf -y install centos-release-advanced-virtualization epel-release dnf-plugins-core
|
||||||
RUN yum -y install https://vitastor.io/rpms/centos/8/vitastor-release-1.0-1.el8.noarch.rpm
|
RUN yum -y install https://vitastor.io/rpms/centos/8/vitastor-release-1.0-1.el8.noarch.rpm
|
||||||
RUN dnf -y install gcc-toolset-9 gcc-toolset-9-gcc-c++ gperftools-devel \
|
RUN dnf --enablerepo='centos-advanced-virtualization' -y install gcc-toolset-9 gcc-toolset-9-gcc-c++ gperftools-devel qemu-kvm fio nodejs rpm-build jerasure-devel gf-complete-devel
|
||||||
fio nodejs rpm-build jerasure-devel gf-complete-devel libibverbs-devel libarchive cmake
|
RUN rm -rf /var/lib/dnf/*; dnf download --disablerepo='*' --enablerepo='vitastor' --source qemu-kvm
|
||||||
RUN dnf download --source fio
|
RUN dnf download --source fio
|
||||||
|
RUN rpm --nomd5 -i qemu*.src.rpm
|
||||||
RUN rpm --nomd5 -i fio*.src.rpm
|
RUN rpm --nomd5 -i fio*.src.rpm
|
||||||
RUN cd ~/rpmbuild/SPECS && dnf builddep -y --enablerepo=powertools --spec fio.spec
|
RUN cd ~/rpmbuild/SPECS && dnf builddep -y --enablerepo=powertools --spec qemu-kvm.spec
|
||||||
|
RUN cd ~/rpmbuild/SPECS && dnf builddep -y --enablerepo=powertools --spec fio.spec && dnf install -y cmake
|
||||||
|
|
||||||
ADD https://vitastor.io/rpms/liburing-el7/liburing-0.7-2.el7.src.rpm /root
|
ADD https://vitastor.io/rpms/liburing-el7/liburing-0.7-2.el7.src.rpm /root
|
||||||
|
|
||||||
|
@ -26,14 +28,14 @@ RUN set -e; \
|
||||||
cp ~/rpmbuild/RPMS/*/liburing* /root/packages/liburing-el8/; \
|
cp ~/rpmbuild/RPMS/*/liburing* /root/packages/liburing-el8/; \
|
||||||
cp ~/rpmbuild/SRPMS/liburing* /root/packages/liburing-el8/
|
cp ~/rpmbuild/SRPMS/liburing* /root/packages/liburing-el8/
|
||||||
|
|
||||||
RUN rpm -i `ls /root/packages/liburing-el8/liburing-*.x86_64.rpm | grep -v debug`
|
RUN rpm -i `ls /root/packages/liburing-el7/liburing-*.x86_64.rpm | grep -v debug`
|
||||||
|
|
||||||
ADD . /root/vitastor
|
ADD . /root/vitastor
|
||||||
|
|
||||||
RUN set -e; \
|
RUN set -e; \
|
||||||
cd /root/vitastor/rpm; \
|
cd /root/vitastor/rpm; \
|
||||||
sh build-tarball.sh; \
|
sh build-tarball.sh; \
|
||||||
cp /root/vitastor-0.6.10.el8.tar.gz ~/rpmbuild/SOURCES; \
|
cp /root/vitastor-0.6.2.el8.tar.gz ~/rpmbuild/SOURCES; \
|
||||||
cp vitastor-el8.spec ~/rpmbuild/SPECS/vitastor.spec; \
|
cp vitastor-el8.spec ~/rpmbuild/SPECS/vitastor.spec; \
|
||||||
cd ~/rpmbuild/SPECS/; \
|
cd ~/rpmbuild/SPECS/; \
|
||||||
rpmbuild -ba vitastor.spec; \
|
rpmbuild -ba vitastor.spec; \
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
Name: vitastor
|
Name: vitastor
|
||||||
Version: 0.6.10
|
Version: 0.6.2
|
||||||
Release: 1%{?dist}
|
Release: 1%{?dist}
|
||||||
Summary: Vitastor, a fast software-defined clustered block storage
|
Summary: Vitastor, a fast software-defined clustered block storage
|
||||||
|
|
||||||
License: Vitastor Network Public License 1.1
|
License: Vitastor Network Public License 1.1
|
||||||
URL: https://vitastor.io/
|
URL: https://vitastor.io/
|
||||||
Source0: vitastor-0.6.10.el8.tar.gz
|
Source0: vitastor-0.6.2.el8.tar.gz
|
||||||
|
|
||||||
BuildRequires: liburing-devel >= 0.6
|
BuildRequires: liburing-devel >= 0.6
|
||||||
BuildRequires: gperftools-devel
|
BuildRequires: gperftools-devel
|
||||||
|
@ -13,13 +13,13 @@ BuildRequires: gcc-toolset-9-gcc-c++
|
||||||
BuildRequires: nodejs >= 10
|
BuildRequires: nodejs >= 10
|
||||||
BuildRequires: jerasure-devel
|
BuildRequires: jerasure-devel
|
||||||
BuildRequires: gf-complete-devel
|
BuildRequires: gf-complete-devel
|
||||||
BuildRequires: libibverbs-devel
|
|
||||||
BuildRequires: cmake
|
BuildRequires: cmake
|
||||||
Requires: vitastor-osd = %{version}-%{release}
|
Requires: fio = 3.7-3.el8
|
||||||
Requires: vitastor-mon = %{version}-%{release}
|
Requires: qemu-kvm = 4.2.0-29.el8.6
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
Requires: nodejs >= 10
|
||||||
Requires: vitastor-client-devel = %{version}-%{release}
|
Requires: liburing >= 0.6
|
||||||
Requires: vitastor-fio = %{version}-%{release}
|
Requires: libJerasure2
|
||||||
|
Requires: lpsolve
|
||||||
|
|
||||||
%description
|
%description
|
||||||
Vitastor is a small, simple and fast clustered block storage (storage for VM drives),
|
Vitastor is a small, simple and fast clustered block storage (storage for VM drives),
|
||||||
|
@ -28,65 +28,13 @@ symmetric clustering and automatic data distribution over any number of drives o
|
||||||
size with configurable redundancy (replication or erasure codes/XOR).
|
size with configurable redundancy (replication or erasure codes/XOR).
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-osd
|
|
||||||
Summary: Vitastor - OSD
|
|
||||||
Requires: libJerasure2
|
|
||||||
Requires: liburing >= 0.6
|
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-osd
|
|
||||||
Vitastor object storage daemon, i.e. server program that stores data.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-mon
|
|
||||||
Summary: Vitastor - monitor
|
|
||||||
Requires: nodejs >= 10
|
|
||||||
Requires: lpsolve
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-mon
|
|
||||||
Vitastor monitor, i.e. server program responsible for watching cluster state and
|
|
||||||
scheduling cluster-level operations.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-client
|
|
||||||
Summary: Vitastor - client
|
|
||||||
Requires: liburing >= 0.6
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-client
|
|
||||||
Vitastor client library and command-line interface.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-client-devel
|
|
||||||
Summary: Vitastor - development files
|
|
||||||
Group: Development/Libraries
|
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-client-devel
|
|
||||||
Vitastor library headers for development.
|
|
||||||
|
|
||||||
|
|
||||||
%package -n vitastor-fio
|
|
||||||
Summary: Vitastor - fio drivers
|
|
||||||
Group: Development/Libraries
|
|
||||||
Requires: vitastor-client = %{version}-%{release}
|
|
||||||
Requires: fio = 3.7-3.el8
|
|
||||||
|
|
||||||
|
|
||||||
%description -n vitastor-fio
|
|
||||||
Vitastor fio drivers for benchmarking.
|
|
||||||
|
|
||||||
|
|
||||||
%prep
|
%prep
|
||||||
%setup -q
|
%setup -q
|
||||||
|
|
||||||
|
|
||||||
%build
|
%build
|
||||||
. /opt/rh/gcc-toolset-9/enable
|
. /opt/rh/gcc-toolset-9/enable
|
||||||
%cmake .
|
%cmake . -DQEMU_PLUGINDIR=qemu-kvm
|
||||||
%make_build
|
%make_build
|
||||||
|
|
||||||
|
|
||||||
|
@ -97,42 +45,22 @@ cd mon
|
||||||
npm install
|
npm install
|
||||||
cd ..
|
cd ..
|
||||||
mkdir -p %buildroot/usr/lib/vitastor
|
mkdir -p %buildroot/usr/lib/vitastor
|
||||||
cp mon/make-osd.sh %buildroot/usr/lib/vitastor
|
|
||||||
cp -r mon %buildroot/usr/lib/vitastor
|
cp -r mon %buildroot/usr/lib/vitastor
|
||||||
|
|
||||||
|
|
||||||
%files
|
%files
|
||||||
%doc GPL-2.0.txt VNPL-1.1.txt README.md README-ru.md
|
%doc
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-osd
|
|
||||||
%_bindir/vitastor-osd
|
|
||||||
%_bindir/vitastor-dump-journal
|
%_bindir/vitastor-dump-journal
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-mon
|
|
||||||
/usr/lib/vitastor/mon
|
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-client
|
|
||||||
%_bindir/vitastor-nbd
|
%_bindir/vitastor-nbd
|
||||||
%_bindir/vitastor-cli
|
%_bindir/vitastor-osd
|
||||||
%_bindir/vitastor-rm
|
%_bindir/vitastor-rm
|
||||||
%_bindir/vita
|
%_libdir/qemu-kvm/block-vitastor.so
|
||||||
%_libdir/libvitastor_blk.so*
|
|
||||||
%_libdir/libvitastor_client.so*
|
|
||||||
/usr/lib/vitastor/make-osd.sh
|
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-client-devel
|
|
||||||
%_includedir/vitastor_c.h
|
|
||||||
%_libdir/pkgconfig
|
|
||||||
|
|
||||||
|
|
||||||
%files -n vitastor-fio
|
|
||||||
%_libdir/libfio_vitastor.so
|
%_libdir/libfio_vitastor.so
|
||||||
%_libdir/libfio_vitastor_blk.so
|
%_libdir/libfio_vitastor_blk.so
|
||||||
%_libdir/libfio_vitastor_sec.so
|
%_libdir/libfio_vitastor_sec.so
|
||||||
|
%_libdir/libvitastor_blk.so
|
||||||
|
%_libdir/libvitastor_client.so
|
||||||
|
/usr/lib/vitastor
|
||||||
|
|
||||||
|
|
||||||
%changelog
|
%changelog
|
||||||
|
|
|
@ -4,8 +4,6 @@ project(vitastor)
|
||||||
|
|
||||||
include(GNUInstallDirs)
|
include(GNUInstallDirs)
|
||||||
|
|
||||||
set(WITH_QEMU false CACHE BOOL "Build QEMU driver inside Vitastor source tree")
|
|
||||||
set(WITH_FIO true CACHE BOOL "Build FIO driver")
|
|
||||||
set(QEMU_PLUGINDIR qemu CACHE STRING "QEMU plugin directory suffix (qemu-kvm on RHEL)")
|
set(QEMU_PLUGINDIR qemu CACHE STRING "QEMU plugin directory suffix (qemu-kvm on RHEL)")
|
||||||
set(WITH_ASAN false CACHE BOOL "Build with AddressSanitizer")
|
set(WITH_ASAN false CACHE BOOL "Build with AddressSanitizer")
|
||||||
if("${CMAKE_INSTALL_PREFIX}" MATCHES "^/usr/local/?$")
|
if("${CMAKE_INSTALL_PREFIX}" MATCHES "^/usr/local/?$")
|
||||||
|
@ -15,8 +13,8 @@ if("${CMAKE_INSTALL_PREFIX}" MATCHES "^/usr/local/?$")
|
||||||
set(CMAKE_INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/${CMAKE_INSTALL_LIBDIR}")
|
set(CMAKE_INSTALL_RPATH "${CMAKE_INSTALL_PREFIX}/${CMAKE_INSTALL_LIBDIR}")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
add_definitions(-DVERSION="0.6.10")
|
add_definitions(-DVERSION="0.6.2")
|
||||||
add_definitions(-Wall -Wno-sign-compare -Wno-comment -Wno-parentheses -Wno-pointer-arith -fdiagnostics-color=always -I ${CMAKE_SOURCE_DIR}/src)
|
add_definitions(-Wall -Wno-sign-compare -Wno-comment -Wno-parentheses -Wno-pointer-arith -I ${CMAKE_SOURCE_DIR}/src)
|
||||||
if (${WITH_ASAN})
|
if (${WITH_ASAN})
|
||||||
add_definitions(-fsanitize=address -fno-omit-frame-pointer)
|
add_definitions(-fsanitize=address -fno-omit-frame-pointer)
|
||||||
add_link_options(-fsanitize=address -fno-omit-frame-pointer)
|
add_link_options(-fsanitize=address -fno-omit-frame-pointer)
|
||||||
|
@ -36,16 +34,9 @@ string(REGEX REPLACE "([\\/\\-]D) *NDEBUG" "" CMAKE_C_FLAGS_RELEASE "${CMAKE_C_F
|
||||||
string(REGEX REPLACE "([\\/\\-]D) *NDEBUG" "" CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL}")
|
string(REGEX REPLACE "([\\/\\-]D) *NDEBUG" "" CMAKE_C_FLAGS_MINSIZEREL "${CMAKE_C_FLAGS_MINSIZEREL}")
|
||||||
string(REGEX REPLACE "([\\/\\-]D) *NDEBUG" "" CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO}")
|
string(REGEX REPLACE "([\\/\\-]D) *NDEBUG" "" CMAKE_C_FLAGS_RELWITHDEBINFO "${CMAKE_C_FLAGS_RELWITHDEBINFO}")
|
||||||
|
|
||||||
macro(install_symlink filepath sympath)
|
|
||||||
install(CODE "execute_process(COMMAND ${CMAKE_COMMAND} -E create_symlink ${filepath} \$ENV{DESTDIR}${sympath})")
|
|
||||||
install(CODE "message(\"-- Created symlink: ${sympath} -> ${filepath}\")")
|
|
||||||
endmacro(install_symlink)
|
|
||||||
|
|
||||||
find_package(PkgConfig)
|
find_package(PkgConfig)
|
||||||
pkg_check_modules(LIBURING REQUIRED liburing)
|
pkg_check_modules(LIBURING REQUIRED liburing)
|
||||||
if (${WITH_QEMU})
|
pkg_check_modules(GLIB REQUIRED glib-2.0)
|
||||||
pkg_check_modules(GLIB REQUIRED glib-2.0)
|
|
||||||
endif (${WITH_QEMU})
|
|
||||||
pkg_check_modules(IBVERBS libibverbs)
|
pkg_check_modules(IBVERBS libibverbs)
|
||||||
if (IBVERBS_LIBRARIES)
|
if (IBVERBS_LIBRARIES)
|
||||||
add_definitions(-DWITH_RDMA)
|
add_definitions(-DWITH_RDMA)
|
||||||
|
@ -71,27 +62,24 @@ target_link_libraries(vitastor_blk
|
||||||
)
|
)
|
||||||
set_target_properties(vitastor_blk PROPERTIES VERSION ${VERSION} SOVERSION 0)
|
set_target_properties(vitastor_blk PROPERTIES VERSION ${VERSION} SOVERSION 0)
|
||||||
|
|
||||||
if (${WITH_FIO})
|
# libfio_vitastor_blk.so
|
||||||
# libfio_vitastor_blk.so
|
add_library(fio_vitastor_blk SHARED
|
||||||
add_library(fio_vitastor_blk SHARED
|
|
||||||
fio_engine.cpp
|
fio_engine.cpp
|
||||||
../json11/json11.cpp
|
../json11/json11.cpp
|
||||||
)
|
)
|
||||||
target_link_libraries(fio_vitastor_blk
|
target_link_libraries(fio_vitastor_blk
|
||||||
vitastor_blk
|
vitastor_blk
|
||||||
)
|
)
|
||||||
endif (${WITH_FIO})
|
|
||||||
|
|
||||||
# libvitastor_common.a
|
# libvitastor_common.a
|
||||||
set(MSGR_RDMA "")
|
|
||||||
if (IBVERBS_LIBRARIES)
|
|
||||||
set(MSGR_RDMA "msgr_rdma.cpp")
|
|
||||||
endif (IBVERBS_LIBRARIES)
|
|
||||||
add_library(vitastor_common STATIC
|
add_library(vitastor_common STATIC
|
||||||
epoll_manager.cpp etcd_state_client.cpp messenger.cpp addr_util.cpp
|
epoll_manager.cpp etcd_state_client.cpp
|
||||||
msgr_stop.cpp msgr_op.cpp msgr_send.cpp msgr_receive.cpp ringloop.cpp ../json11/json11.cpp
|
messenger.cpp msgr_stop.cpp msgr_op.cpp msgr_send.cpp msgr_receive.cpp ringloop.cpp ../json11/json11.cpp
|
||||||
http_client.cpp osd_ops.cpp pg_states.cpp timerfd_manager.cpp base64.cpp ${MSGR_RDMA}
|
http_client.cpp osd_ops.cpp pg_states.cpp timerfd_manager.cpp base64.cpp
|
||||||
)
|
)
|
||||||
|
if (IBVERBS_LIBRARIES)
|
||||||
|
target_sources(vitastor_common PRIVATE msgr_rdma.cpp)
|
||||||
|
endif (IBVERBS_LIBRARIES)
|
||||||
target_compile_options(vitastor_common PUBLIC -fPIC)
|
target_compile_options(vitastor_common PUBLIC -fPIC)
|
||||||
|
|
||||||
# vitastor-osd
|
# vitastor-osd
|
||||||
|
@ -107,25 +95,19 @@ target_link_libraries(vitastor-osd
|
||||||
${IBVERBS_LIBRARIES}
|
${IBVERBS_LIBRARIES}
|
||||||
)
|
)
|
||||||
|
|
||||||
if (${WITH_FIO})
|
# libfio_vitastor_sec.so
|
||||||
# libfio_vitastor_sec.so
|
add_library(fio_vitastor_sec SHARED
|
||||||
add_library(fio_vitastor_sec SHARED
|
|
||||||
fio_sec_osd.cpp
|
fio_sec_osd.cpp
|
||||||
rw_blocking.cpp
|
rw_blocking.cpp
|
||||||
addr_util.cpp
|
)
|
||||||
)
|
target_link_libraries(fio_vitastor_sec
|
||||||
target_link_libraries(fio_vitastor_sec
|
|
||||||
tcmalloc_minimal
|
tcmalloc_minimal
|
||||||
)
|
)
|
||||||
endif (${WITH_FIO})
|
|
||||||
|
|
||||||
# libvitastor_client.so
|
# libvitastor_client.so
|
||||||
add_library(vitastor_client SHARED
|
add_library(vitastor_client SHARED
|
||||||
cluster_client.cpp
|
cluster_client.cpp
|
||||||
cluster_client_list.cpp
|
|
||||||
vitastor_c.cpp
|
|
||||||
)
|
)
|
||||||
set_target_properties(vitastor_client PROPERTIES PUBLIC_HEADER "vitastor_c.h")
|
|
||||||
target_link_libraries(vitastor_client
|
target_link_libraries(vitastor_client
|
||||||
vitastor_common
|
vitastor_common
|
||||||
tcmalloc_minimal
|
tcmalloc_minimal
|
||||||
|
@ -134,15 +116,13 @@ target_link_libraries(vitastor_client
|
||||||
)
|
)
|
||||||
set_target_properties(vitastor_client PROPERTIES VERSION ${VERSION} SOVERSION 0)
|
set_target_properties(vitastor_client PROPERTIES VERSION ${VERSION} SOVERSION 0)
|
||||||
|
|
||||||
if (${WITH_FIO})
|
# libfio_vitastor.so
|
||||||
# libfio_vitastor.so
|
add_library(fio_vitastor SHARED
|
||||||
add_library(fio_vitastor SHARED
|
|
||||||
fio_cluster.cpp
|
fio_cluster.cpp
|
||||||
)
|
)
|
||||||
target_link_libraries(fio_vitastor
|
target_link_libraries(fio_vitastor
|
||||||
vitastor_client
|
vitastor_client
|
||||||
)
|
)
|
||||||
endif (${WITH_FIO})
|
|
||||||
|
|
||||||
# vitastor-nbd
|
# vitastor-nbd
|
||||||
add_executable(vitastor-nbd
|
add_executable(vitastor-nbd
|
||||||
|
@ -152,49 +132,49 @@ target_link_libraries(vitastor-nbd
|
||||||
vitastor_client
|
vitastor_client
|
||||||
)
|
)
|
||||||
|
|
||||||
# vitastor-cli
|
# vitastor-rm
|
||||||
add_executable(vitastor-cli
|
add_executable(vitastor-rm
|
||||||
cli.cpp cli_alloc_osd.cpp cli_simple_offsets.cpp cli_df.cpp
|
rm_inode.cpp
|
||||||
cli_ls.cpp cli_create.cpp cli_modify.cpp cli_flatten.cpp cli_merge.cpp cli_rm.cpp cli_snap_rm.cpp
|
|
||||||
)
|
)
|
||||||
target_link_libraries(vitastor-cli
|
target_link_libraries(vitastor-rm
|
||||||
vitastor_client
|
vitastor_client
|
||||||
)
|
)
|
||||||
configure_file(vitastor.pc.in vitastor.pc @ONLY)
|
|
||||||
|
|
||||||
# vitastor-dump-journal
|
# vitastor-dump-journal
|
||||||
add_executable(vitastor-dump-journal
|
add_executable(vitastor-dump-journal
|
||||||
dump_journal.cpp crc32c.c
|
dump_journal.cpp crc32c.c
|
||||||
)
|
)
|
||||||
|
|
||||||
if (${WITH_QEMU})
|
# qemu_driver.so
|
||||||
# qemu_driver.so
|
add_library(qemu_proxy STATIC qemu_proxy.cpp)
|
||||||
add_library(qemu_vitastor SHARED
|
target_compile_options(qemu_proxy PUBLIC -fPIC)
|
||||||
qemu_driver.c
|
target_include_directories(qemu_proxy PUBLIC
|
||||||
)
|
|
||||||
target_compile_options(qemu_vitastor PUBLIC -DVITASTOR_SOURCE_TREE)
|
|
||||||
target_include_directories(qemu_vitastor PUBLIC
|
|
||||||
../qemu/b/qemu
|
../qemu/b/qemu
|
||||||
../qemu/include
|
../qemu/include
|
||||||
${GLIB_INCLUDE_DIRS}
|
${GLIB_INCLUDE_DIRS}
|
||||||
)
|
)
|
||||||
target_link_libraries(qemu_vitastor
|
target_link_libraries(qemu_proxy
|
||||||
vitastor_client
|
vitastor_client
|
||||||
)
|
)
|
||||||
set_target_properties(qemu_vitastor PROPERTIES
|
add_library(qemu_vitastor SHARED
|
||||||
|
qemu_driver.c
|
||||||
|
)
|
||||||
|
target_link_libraries(qemu_vitastor
|
||||||
|
qemu_proxy
|
||||||
|
)
|
||||||
|
set_target_properties(qemu_vitastor PROPERTIES
|
||||||
PREFIX ""
|
PREFIX ""
|
||||||
OUTPUT_NAME "block-vitastor"
|
OUTPUT_NAME "block-vitastor"
|
||||||
)
|
)
|
||||||
endif (${WITH_QEMU})
|
|
||||||
|
|
||||||
### Test stubs
|
### Test stubs
|
||||||
|
|
||||||
# stub_osd, stub_bench, osd_test
|
# stub_osd, stub_bench, osd_test
|
||||||
add_executable(stub_osd stub_osd.cpp rw_blocking.cpp addr_util.cpp)
|
add_executable(stub_osd stub_osd.cpp rw_blocking.cpp)
|
||||||
target_link_libraries(stub_osd tcmalloc_minimal)
|
target_link_libraries(stub_osd tcmalloc_minimal)
|
||||||
add_executable(stub_bench stub_bench.cpp rw_blocking.cpp addr_util.cpp)
|
add_executable(stub_bench stub_bench.cpp rw_blocking.cpp)
|
||||||
target_link_libraries(stub_bench tcmalloc_minimal)
|
target_link_libraries(stub_bench tcmalloc_minimal)
|
||||||
add_executable(osd_test osd_test.cpp rw_blocking.cpp addr_util.cpp)
|
add_executable(osd_test osd_test.cpp rw_blocking.cpp)
|
||||||
target_link_libraries(osd_test tcmalloc_minimal)
|
target_link_libraries(osd_test tcmalloc_minimal)
|
||||||
|
|
||||||
# osd_rmw_test
|
# osd_rmw_test
|
||||||
|
@ -219,18 +199,10 @@ target_link_libraries(osd_peering_pg_test tcmalloc_minimal)
|
||||||
# test_allocator
|
# test_allocator
|
||||||
add_executable(test_allocator test_allocator.cpp allocator.cpp)
|
add_executable(test_allocator test_allocator.cpp allocator.cpp)
|
||||||
|
|
||||||
# test_cas
|
|
||||||
add_executable(test_cas
|
|
||||||
test_cas.cpp
|
|
||||||
)
|
|
||||||
target_link_libraries(test_cas
|
|
||||||
vitastor_client
|
|
||||||
)
|
|
||||||
|
|
||||||
# test_cluster_client
|
# test_cluster_client
|
||||||
add_executable(test_cluster_client
|
add_executable(test_cluster_client
|
||||||
test_cluster_client.cpp
|
test_cluster_client.cpp
|
||||||
pg_states.cpp osd_ops.cpp cluster_client.cpp cluster_client_list.cpp msgr_op.cpp mock/messenger.cpp msgr_stop.cpp
|
pg_states.cpp osd_ops.cpp cluster_client.cpp msgr_op.cpp mock/messenger.cpp msgr_stop.cpp
|
||||||
etcd_state_client.cpp timerfd_manager.cpp ../json11/json11.cpp
|
etcd_state_client.cpp timerfd_manager.cpp ../json11/json11.cpp
|
||||||
)
|
)
|
||||||
target_compile_definitions(test_cluster_client PUBLIC -D__MOCK__)
|
target_compile_definitions(test_cluster_client PUBLIC -D__MOCK__)
|
||||||
|
@ -244,18 +216,6 @@ target_include_directories(test_cluster_client PUBLIC ${CMAKE_SOURCE_DIR}/src/mo
|
||||||
|
|
||||||
### Install
|
### Install
|
||||||
|
|
||||||
install(TARGETS vitastor-osd vitastor-dump-journal vitastor-nbd vitastor-cli RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR})
|
install(TARGETS vitastor-osd vitastor-dump-journal vitastor-nbd vitastor-rm RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR})
|
||||||
install_symlink(vitastor-cli ${CMAKE_INSTALL_PREFIX}/${CMAKE_INSTALL_BINDIR}/vitastor-rm)
|
install(TARGETS fio_vitastor fio_vitastor_blk fio_vitastor_sec vitastor_blk vitastor_client LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR})
|
||||||
install_symlink(vitastor-cli ${CMAKE_INSTALL_PREFIX}/${CMAKE_INSTALL_BINDIR}/vita)
|
install(TARGETS qemu_vitastor LIBRARY DESTINATION /usr/${CMAKE_INSTALL_LIBDIR}/${QEMU_PLUGINDIR})
|
||||||
install(
|
|
||||||
TARGETS vitastor_blk vitastor_client
|
|
||||||
LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR}
|
|
||||||
PUBLIC_HEADER DESTINATION ${CMAKE_INSTALL_INCLUDEDIR}
|
|
||||||
)
|
|
||||||
install(FILES ${CMAKE_CURRENT_BINARY_DIR}/vitastor.pc DESTINATION ${CMAKE_INSTALL_LIBDIR}/pkgconfig)
|
|
||||||
if (${WITH_FIO})
|
|
||||||
install(TARGETS fio_vitastor fio_vitastor_blk fio_vitastor_sec LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR})
|
|
||||||
endif (${WITH_FIO})
|
|
||||||
if (${WITH_QEMU})
|
|
||||||
install(TARGETS qemu_vitastor LIBRARY DESTINATION /usr/${CMAKE_INSTALL_LIBDIR}/${QEMU_PLUGINDIR})
|
|
||||||
endif (${WITH_QEMU})
|
|
||||||
|
|
|
@ -1,60 +0,0 @@
|
||||||
#include <arpa/inet.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <stdio.h>
|
|
||||||
|
|
||||||
#include <stdexcept>
|
|
||||||
|
|
||||||
#include "addr_util.h"
|
|
||||||
|
|
||||||
bool string_to_addr(std::string str, bool parse_port, int default_port, struct sockaddr *addr)
|
|
||||||
{
|
|
||||||
if (parse_port)
|
|
||||||
{
|
|
||||||
int p = str.rfind(':');
|
|
||||||
if (p != std::string::npos && !(str.length() > 0 && str[p-1] == ']')) // "[ipv6]" which contains ':'
|
|
||||||
{
|
|
||||||
char null_byte = 0;
|
|
||||||
int n = sscanf(str.c_str()+p+1, "%d%c", &default_port, &null_byte);
|
|
||||||
if (n != 1 || default_port >= 0x10000)
|
|
||||||
return false;
|
|
||||||
str = str.substr(0, p);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (inet_pton(AF_INET, str.c_str(), &((struct sockaddr_in*)addr)->sin_addr) == 1)
|
|
||||||
{
|
|
||||||
addr->sa_family = AF_INET;
|
|
||||||
((struct sockaddr_in*)addr)->sin_port = htons(default_port);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (str.length() >= 2 && str[0] == '[' && str[str.length()-1] == ']')
|
|
||||||
str = str.substr(1, str.length()-2);
|
|
||||||
if (inet_pton(AF_INET6, str.c_str(), &((struct sockaddr_in6*)addr)->sin6_addr) == 1)
|
|
||||||
{
|
|
||||||
addr->sa_family = AF_INET6;
|
|
||||||
((struct sockaddr_in6*)addr)->sin6_port = htons(default_port);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string addr_to_string(const sockaddr &addr)
|
|
||||||
{
|
|
||||||
char peer_str[256];
|
|
||||||
bool ok = false;
|
|
||||||
int port;
|
|
||||||
if (addr.sa_family == AF_INET)
|
|
||||||
{
|
|
||||||
ok = !!inet_ntop(AF_INET, &((sockaddr_in*)&addr)->sin_addr, peer_str, 256);
|
|
||||||
port = ntohs(((sockaddr_in*)&addr)->sin_port);
|
|
||||||
}
|
|
||||||
else if (addr.sa_family == AF_INET6)
|
|
||||||
{
|
|
||||||
ok = !!inet_ntop(AF_INET6, &((sockaddr_in6*)&addr)->sin6_addr, peer_str, 256);
|
|
||||||
port = ntohs(((sockaddr_in6*)&addr)->sin6_port);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
throw std::runtime_error("Unknown address family "+std::to_string(addr.sa_family));
|
|
||||||
if (!ok)
|
|
||||||
throw std::runtime_error(std::string("inet_ntop: ") + strerror(errno));
|
|
||||||
return std::string(peer_str)+":"+std::to_string(port);
|
|
||||||
}
|
|
|
@ -1,7 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <sys/socket.h>
|
|
||||||
#include <string>
|
|
||||||
|
|
||||||
bool string_to_addr(std::string str, bool parse_port, int default_port, struct sockaddr *addr);
|
|
||||||
std::string addr_to_string(const sockaddr &addr);
|
|
|
@ -43,16 +43,16 @@ int blockstore_t::read_bitmap(object_id oid, uint64_t target_version, void *bitm
|
||||||
return impl->read_bitmap(oid, target_version, bitmap, result_version);
|
return impl->read_bitmap(oid, target_version, bitmap, result_version);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::unordered_map<object_id, uint64_t> & blockstore_t::get_unstable_writes()
|
||||||
|
{
|
||||||
|
return impl->unstable_writes;
|
||||||
|
}
|
||||||
|
|
||||||
std::map<uint64_t, uint64_t> & blockstore_t::get_inode_space_stats()
|
std::map<uint64_t, uint64_t> & blockstore_t::get_inode_space_stats()
|
||||||
{
|
{
|
||||||
return impl->inode_space_stats;
|
return impl->inode_space_stats;
|
||||||
}
|
}
|
||||||
|
|
||||||
void blockstore_t::dump_diagnostics()
|
|
||||||
{
|
|
||||||
return impl->dump_diagnostics();
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t blockstore_t::get_block_size()
|
uint32_t blockstore_t::get_block_size()
|
||||||
{
|
{
|
||||||
return impl->get_block_size();
|
return impl->get_block_size();
|
||||||
|
@ -68,11 +68,6 @@ uint64_t blockstore_t::get_free_block_count()
|
||||||
return impl->get_free_block_count();
|
return impl->get_free_block_count();
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t blockstore_t::get_journal_size()
|
|
||||||
{
|
|
||||||
return impl->get_journal_size();
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t blockstore_t::get_bitmap_granularity()
|
uint32_t blockstore_t::get_bitmap_granularity()
|
||||||
{
|
{
|
||||||
return impl->get_bitmap_granularity();
|
return impl->get_bitmap_granularity();
|
||||||
|
|
|
@ -183,18 +183,16 @@ public:
|
||||||
// Simplified synchronous operation: get object bitmap & current version
|
// Simplified synchronous operation: get object bitmap & current version
|
||||||
int read_bitmap(object_id oid, uint64_t target_version, void *bitmap, uint64_t *result_version = NULL);
|
int read_bitmap(object_id oid, uint64_t target_version, void *bitmap, uint64_t *result_version = NULL);
|
||||||
|
|
||||||
|
// Unstable writes are added here (map of object_id -> version)
|
||||||
|
std::unordered_map<object_id, uint64_t> & get_unstable_writes();
|
||||||
|
|
||||||
// Get per-inode space usage statistics
|
// Get per-inode space usage statistics
|
||||||
std::map<uint64_t, uint64_t> & get_inode_space_stats();
|
std::map<uint64_t, uint64_t> & get_inode_space_stats();
|
||||||
|
|
||||||
// Print diagnostics to stdout
|
|
||||||
void dump_diagnostics();
|
|
||||||
|
|
||||||
// FIXME rename to object_size
|
// FIXME rename to object_size
|
||||||
uint32_t get_block_size();
|
uint32_t get_block_size();
|
||||||
uint64_t get_block_count();
|
uint64_t get_block_count();
|
||||||
uint64_t get_free_block_count();
|
uint64_t get_free_block_count();
|
||||||
|
|
||||||
uint64_t get_journal_size();
|
|
||||||
|
|
||||||
uint32_t get_bitmap_granularity();
|
uint32_t get_bitmap_granularity();
|
||||||
};
|
};
|
||||||
|
|
|
@ -182,75 +182,6 @@ void journal_flusher_t::release_trim()
|
||||||
trim_wanted--;
|
trim_wanted--;
|
||||||
}
|
}
|
||||||
|
|
||||||
void journal_flusher_t::dump_diagnostics()
|
|
||||||
{
|
|
||||||
const char *unflushable_type = "";
|
|
||||||
obj_ver_id unflushable = { 0 };
|
|
||||||
// Try to find out if there is a flushable object for information
|
|
||||||
for (object_id cur_oid: flush_queue)
|
|
||||||
{
|
|
||||||
obj_ver_id cur = { .oid = cur_oid, .version = flush_versions[cur_oid] };
|
|
||||||
auto dirty_end = bs->dirty_db.find(cur);
|
|
||||||
if (dirty_end == bs->dirty_db.end())
|
|
||||||
{
|
|
||||||
// Already flushed
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
auto repeat_it = sync_to_repeat.find(cur.oid);
|
|
||||||
if (repeat_it != sync_to_repeat.end())
|
|
||||||
{
|
|
||||||
// Someone is already flushing it
|
|
||||||
unflushable_type = "locked,";
|
|
||||||
unflushable = cur;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (dirty_end->second.journal_sector >= bs->journal.dirty_start &&
|
|
||||||
(bs->journal.dirty_start >= bs->journal.used_start ||
|
|
||||||
dirty_end->second.journal_sector < bs->journal.used_start))
|
|
||||||
{
|
|
||||||
// Object is more recent than possible to flush
|
|
||||||
bool found = try_find_older(dirty_end, cur);
|
|
||||||
if (!found)
|
|
||||||
{
|
|
||||||
unflushable_type = "dirty,";
|
|
||||||
unflushable = cur;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
unflushable_type = "ok,";
|
|
||||||
unflushable = cur;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
printf(
|
|
||||||
"Flusher: queued=%ld first=%s%lx:%lx trim_wanted=%d dequeuing=%d trimming=%d cur=%d target=%d active=%d syncing=%d\n",
|
|
||||||
flush_queue.size(), unflushable_type, unflushable.oid.inode, unflushable.oid.stripe,
|
|
||||||
trim_wanted, dequeuing, trimming, cur_flusher_count, target_flusher_count,
|
|
||||||
active_flushers, syncing_flushers
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool journal_flusher_t::try_find_older(std::map<obj_ver_id, dirty_entry>::iterator & dirty_end, obj_ver_id & cur)
|
|
||||||
{
|
|
||||||
bool found = false;
|
|
||||||
while (dirty_end != bs->dirty_db.begin())
|
|
||||||
{
|
|
||||||
dirty_end--;
|
|
||||||
if (dirty_end->first.oid != cur.oid)
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (!(dirty_end->second.journal_sector >= bs->journal.dirty_start &&
|
|
||||||
(bs->journal.dirty_start >= bs->journal.used_start ||
|
|
||||||
dirty_end->second.journal_sector < bs->journal.used_start)))
|
|
||||||
{
|
|
||||||
found = true;
|
|
||||||
cur.version = dirty_end->first.version;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return found;
|
|
||||||
}
|
|
||||||
|
|
||||||
#define await_sqe(label) \
|
#define await_sqe(label) \
|
||||||
resume_##label:\
|
resume_##label:\
|
||||||
sqe = bs->get_sqe();\
|
sqe = bs->get_sqe();\
|
||||||
|
@ -355,15 +286,30 @@ stop_flusher:
|
||||||
// And it may even block writes if we don't flush the older version
|
// And it may even block writes if we don't flush the older version
|
||||||
// (if it's in the beginning of the journal)...
|
// (if it's in the beginning of the journal)...
|
||||||
// So first try to find an older version of the same object to flush.
|
// So first try to find an older version of the same object to flush.
|
||||||
bool found = flusher->try_find_older(dirty_end, cur);
|
bool found = false;
|
||||||
|
while (dirty_end != bs->dirty_db.begin())
|
||||||
|
{
|
||||||
|
dirty_end--;
|
||||||
|
if (dirty_end->first.oid != cur.oid)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (!(dirty_end->second.journal_sector >= bs->journal.dirty_start &&
|
||||||
|
(bs->journal.dirty_start >= bs->journal.used_start ||
|
||||||
|
dirty_end->second.journal_sector < bs->journal.used_start)))
|
||||||
|
{
|
||||||
|
found = true;
|
||||||
|
cur.version = dirty_end->first.version;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
if (!found)
|
if (!found)
|
||||||
{
|
{
|
||||||
// Try other objects
|
// Try other objects
|
||||||
flusher->sync_to_repeat.erase(cur.oid);
|
flusher->sync_to_repeat.erase(cur.oid);
|
||||||
int search_left = flusher->flush_queue.size() - 1;
|
int search_left = flusher->flush_queue.size() - 1;
|
||||||
#ifdef BLOCKSTORE_DEBUG
|
#ifdef BLOCKSTORE_DEBUG
|
||||||
printf("Flusher overran writers (%lx:%lx v%lu, dirty_start=%08lx) - searching for older flushes (%d left)\n",
|
printf("Flusher overran writers (dirty_start=%08lx) - searching for older flushes (%d left)\n", bs->journal.dirty_start, search_left);
|
||||||
cur.oid.inode, cur.oid.stripe, cur.version, bs->journal.dirty_start, search_left);
|
|
||||||
#endif
|
#endif
|
||||||
while (search_left > 0)
|
while (search_left > 0)
|
||||||
{
|
{
|
||||||
|
@ -386,12 +332,7 @@ stop_flusher:
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
repeat_it = flusher->sync_to_repeat.find(cur.oid);
|
repeat_it = flusher->sync_to_repeat.find(cur.oid);
|
||||||
if (repeat_it != flusher->sync_to_repeat.end())
|
if (repeat_it == flusher->sync_to_repeat.end())
|
||||||
{
|
|
||||||
if (repeat_it->second < cur.version)
|
|
||||||
repeat_it->second = cur.version;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
{
|
||||||
flusher->sync_to_repeat[cur.oid] = 0;
|
flusher->sync_to_repeat[cur.oid] = 0;
|
||||||
break;
|
break;
|
||||||
|
@ -547,9 +488,8 @@ resume_1:
|
||||||
clean_disk_entry *new_entry = (clean_disk_entry*)(meta_new.buf + meta_new.pos*bs->clean_entry_size);
|
clean_disk_entry *new_entry = (clean_disk_entry*)(meta_new.buf + meta_new.pos*bs->clean_entry_size);
|
||||||
if (new_entry->oid.inode != 0 && new_entry->oid != cur.oid)
|
if (new_entry->oid.inode != 0 && new_entry->oid != cur.oid)
|
||||||
{
|
{
|
||||||
printf("Fatal error (metadata corruption or bug): tried to delete metadata entry %lu (%lx:%lx v%lu) while deleting %lx:%lx\n",
|
printf("Fatal error (metadata corruption or bug): tried to delete metadata entry %lu (%lx:%lx) while deleting %lx:%lx\n",
|
||||||
clean_loc >> bs->block_order, new_entry->oid.inode, new_entry->oid.stripe,
|
clean_loc >> bs->block_order, new_entry->oid.inode, new_entry->oid.stripe, cur.oid.inode, cur.oid.stripe);
|
||||||
new_entry->version, cur.oid.inode, cur.oid.stripe);
|
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
// zero out new metadata entry
|
// zero out new metadata entry
|
||||||
|
@ -560,9 +500,8 @@ resume_1:
|
||||||
clean_disk_entry *new_entry = (clean_disk_entry*)(meta_new.buf + meta_new.pos*bs->clean_entry_size);
|
clean_disk_entry *new_entry = (clean_disk_entry*)(meta_new.buf + meta_new.pos*bs->clean_entry_size);
|
||||||
if (new_entry->oid.inode != 0 && new_entry->oid != cur.oid)
|
if (new_entry->oid.inode != 0 && new_entry->oid != cur.oid)
|
||||||
{
|
{
|
||||||
printf("Fatal error (metadata corruption or bug): tried to overwrite non-zero metadata entry %lu (%lx:%lx v%lu) with %lx:%lx v%lu\n",
|
printf("Fatal error (metadata corruption or bug): tried to overwrite non-zero metadata entry %lu (%lx:%lx) with %lx:%lx\n",
|
||||||
clean_loc >> bs->block_order, new_entry->oid.inode, new_entry->oid.stripe, new_entry->version,
|
clean_loc >> bs->block_order, new_entry->oid.inode, new_entry->oid.stripe, cur.oid.inode, cur.oid.stripe);
|
||||||
cur.oid.inode, cur.oid.stripe, cur.version);
|
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
new_entry->oid = cur.oid;
|
new_entry->oid = cur.oid;
|
||||||
|
@ -745,15 +684,12 @@ bool journal_flusher_co::scan_dirty(int wait_base)
|
||||||
offset = dirty_it->second.offset;
|
offset = dirty_it->second.offset;
|
||||||
end_offset = dirty_it->second.offset + dirty_it->second.len;
|
end_offset = dirty_it->second.offset + dirty_it->second.len;
|
||||||
it = v.begin();
|
it = v.begin();
|
||||||
while (end_offset > offset)
|
while (1)
|
||||||
{
|
{
|
||||||
for (; it != v.end(); it++)
|
for (; it != v.end(); it++)
|
||||||
if (it->offset+it->len > offset)
|
if (it->offset >= offset)
|
||||||
break;
|
break;
|
||||||
// If all items end before offset or if the found item starts after end_offset, just insert the buffer
|
if (it == v.end() || it->offset > offset && it->len > 0)
|
||||||
// If (offset < it->offset < end_offset) insert (offset..it->offset) part
|
|
||||||
// If (it->offset <= offset <= it->offset+it->len) then just skip to it->offset+it->len
|
|
||||||
if (it == v.end() || it->offset > offset)
|
|
||||||
{
|
{
|
||||||
submit_offset = dirty_it->second.location + offset - dirty_it->second.offset;
|
submit_offset = dirty_it->second.location + offset - dirty_it->second.offset;
|
||||||
submit_len = it == v.end() || it->offset >= end_offset ? end_offset-offset : it->offset-offset;
|
submit_len = it == v.end() || it->offset >= end_offset ? end_offset-offset : it->offset-offset;
|
||||||
|
@ -777,7 +713,7 @@ bool journal_flusher_co::scan_dirty(int wait_base)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
offset = it->offset+it->len;
|
offset = it->offset+it->len;
|
||||||
if (it == v.end())
|
if (it == v.end() || offset >= end_offset)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -97,9 +97,6 @@ class journal_flusher_t
|
||||||
std::map<uint64_t, meta_sector_t> meta_sectors;
|
std::map<uint64_t, meta_sector_t> meta_sectors;
|
||||||
std::deque<object_id> flush_queue;
|
std::deque<object_id> flush_queue;
|
||||||
std::map<object_id, uint64_t> flush_versions;
|
std::map<object_id, uint64_t> flush_versions;
|
||||||
|
|
||||||
bool try_find_older(std::map<obj_ver_id, dirty_entry>::iterator & dirty_end, obj_ver_id & cur);
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
journal_flusher_t(blockstore_impl_t *bs);
|
journal_flusher_t(blockstore_impl_t *bs);
|
||||||
~journal_flusher_t();
|
~journal_flusher_t();
|
||||||
|
@ -111,5 +108,4 @@ public:
|
||||||
void enqueue_flush(obj_ver_id oid);
|
void enqueue_flush(obj_ver_id oid);
|
||||||
void unshift_flush(obj_ver_id oid, bool force);
|
void unshift_flush(obj_ver_id oid, bool force);
|
||||||
void remove_flush(object_id oid);
|
void remove_flush(object_id oid);
|
||||||
void dump_diagnostics();
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -235,12 +235,6 @@ void blockstore_impl_t::loop()
|
||||||
{
|
{
|
||||||
throw std::runtime_error(std::string("io_uring_submit: ") + strerror(-ret));
|
throw std::runtime_error(std::string("io_uring_submit: ") + strerror(-ret));
|
||||||
}
|
}
|
||||||
for (auto s: journal.submitting_sectors)
|
|
||||||
{
|
|
||||||
// Mark journal sector writes as submitted
|
|
||||||
journal.sector_info[s].submit_id = 0;
|
|
||||||
}
|
|
||||||
journal.submitting_sectors.clear();
|
|
||||||
if ((initial_ring_space - ringloop->space_left()) > 0)
|
if ((initial_ring_space - ringloop->space_left()) > 0)
|
||||||
{
|
{
|
||||||
live = true;
|
live = true;
|
||||||
|
@ -601,9 +595,3 @@ void blockstore_impl_t::process_list(blockstore_op_t *op)
|
||||||
op->buf = stable;
|
op->buf = stable;
|
||||||
FINISH_OP(op);
|
FINISH_OP(op);
|
||||||
}
|
}
|
||||||
|
|
||||||
void blockstore_impl_t::dump_diagnostics()
|
|
||||||
{
|
|
||||||
journal.dump_diagnostics();
|
|
||||||
flusher->dump_diagnostics();
|
|
||||||
}
|
|
||||||
|
|
|
@ -54,14 +54,6 @@
|
||||||
#define IS_BIG_WRITE(st) (((st) & 0x0F) == BS_ST_BIG_WRITE)
|
#define IS_BIG_WRITE(st) (((st) & 0x0F) == BS_ST_BIG_WRITE)
|
||||||
#define IS_DELETE(st) (((st) & 0x0F) == BS_ST_DELETE)
|
#define IS_DELETE(st) (((st) & 0x0F) == BS_ST_DELETE)
|
||||||
|
|
||||||
#define BS_SUBMIT_CHECK_SQES(n) \
|
|
||||||
if (ringloop->space_left() < (n))\
|
|
||||||
{\
|
|
||||||
/* Pause until there are more requests available */\
|
|
||||||
PRIV(op)->wait_for = WAIT_SQE;\
|
|
||||||
return 0;\
|
|
||||||
}
|
|
||||||
|
|
||||||
#define BS_SUBMIT_GET_SQE(sqe, data) \
|
#define BS_SUBMIT_GET_SQE(sqe, data) \
|
||||||
BS_SUBMIT_GET_ONLY_SQE(sqe); \
|
BS_SUBMIT_GET_ONLY_SQE(sqe); \
|
||||||
struct ring_data_t *data = ((ring_data_t*)sqe->user_data)
|
struct ring_data_t *data = ((ring_data_t*)sqe->user_data)
|
||||||
|
@ -178,7 +170,7 @@ struct blockstore_op_private_t
|
||||||
std::vector<fulfill_read_t> read_vec;
|
std::vector<fulfill_read_t> read_vec;
|
||||||
|
|
||||||
// Sync, write
|
// Sync, write
|
||||||
int min_flushed_journal_sector, max_flushed_journal_sector;
|
uint64_t min_flushed_journal_sector, max_flushed_journal_sector;
|
||||||
|
|
||||||
// Write
|
// Write
|
||||||
struct iovec iov_zerofill[3];
|
struct iovec iov_zerofill[3];
|
||||||
|
@ -259,7 +251,6 @@ class blockstore_impl_t
|
||||||
int data_fd;
|
int data_fd;
|
||||||
uint64_t meta_size, meta_area, meta_len;
|
uint64_t meta_size, meta_area, meta_len;
|
||||||
uint64_t data_size, data_len;
|
uint64_t data_size, data_len;
|
||||||
uint64_t data_device_sect, meta_device_sect, journal_device_sect;
|
|
||||||
|
|
||||||
void *metadata_buffer = NULL;
|
void *metadata_buffer = NULL;
|
||||||
|
|
||||||
|
@ -291,10 +282,6 @@ class blockstore_impl_t
|
||||||
void open_journal();
|
void open_journal();
|
||||||
uint8_t* get_clean_entry_bitmap(uint64_t block_loc, int offset);
|
uint8_t* get_clean_entry_bitmap(uint64_t block_loc, int offset);
|
||||||
|
|
||||||
// Journaling
|
|
||||||
void prepare_journal_sector_write(int sector, blockstore_op_t *op);
|
|
||||||
void handle_journal_write(ring_data_t *data, uint64_t flush_id);
|
|
||||||
|
|
||||||
// Asynchronous init
|
// Asynchronous init
|
||||||
int initialized;
|
int initialized;
|
||||||
int metadata_buf_size;
|
int metadata_buf_size;
|
||||||
|
@ -322,18 +309,21 @@ class blockstore_impl_t
|
||||||
|
|
||||||
// Sync
|
// Sync
|
||||||
int continue_sync(blockstore_op_t *op, bool queue_has_in_progress_sync);
|
int continue_sync(blockstore_op_t *op, bool queue_has_in_progress_sync);
|
||||||
|
void handle_sync_event(ring_data_t *data, blockstore_op_t *op);
|
||||||
void ack_sync(blockstore_op_t *op);
|
void ack_sync(blockstore_op_t *op);
|
||||||
|
|
||||||
// Stabilize
|
// Stabilize
|
||||||
int dequeue_stable(blockstore_op_t *op);
|
int dequeue_stable(blockstore_op_t *op);
|
||||||
int continue_stable(blockstore_op_t *op);
|
int continue_stable(blockstore_op_t *op);
|
||||||
void mark_stable(const obj_ver_id & ov, bool forget_dirty = false);
|
void mark_stable(const obj_ver_id & ov, bool forget_dirty = false);
|
||||||
|
void handle_stable_event(ring_data_t *data, blockstore_op_t *op);
|
||||||
void stabilize_object(object_id oid, uint64_t max_ver);
|
void stabilize_object(object_id oid, uint64_t max_ver);
|
||||||
|
|
||||||
// Rollback
|
// Rollback
|
||||||
int dequeue_rollback(blockstore_op_t *op);
|
int dequeue_rollback(blockstore_op_t *op);
|
||||||
int continue_rollback(blockstore_op_t *op);
|
int continue_rollback(blockstore_op_t *op);
|
||||||
void mark_rolled_back(const obj_ver_id & ov);
|
void mark_rolled_back(const obj_ver_id & ov);
|
||||||
|
void handle_rollback_event(ring_data_t *data, blockstore_op_t *op);
|
||||||
void erase_dirty(blockstore_dirty_db_t::iterator dirty_start, blockstore_dirty_db_t::iterator dirty_end, uint64_t clean_loc);
|
void erase_dirty(blockstore_dirty_db_t::iterator dirty_start, blockstore_dirty_db_t::iterator dirty_end, uint64_t clean_loc);
|
||||||
|
|
||||||
// List
|
// List
|
||||||
|
@ -371,12 +361,8 @@ public:
|
||||||
// Space usage statistics
|
// Space usage statistics
|
||||||
std::map<uint64_t, uint64_t> inode_space_stats;
|
std::map<uint64_t, uint64_t> inode_space_stats;
|
||||||
|
|
||||||
// Print diagnostics to stdout
|
|
||||||
void dump_diagnostics();
|
|
||||||
|
|
||||||
inline uint32_t get_block_size() { return block_size; }
|
inline uint32_t get_block_size() { return block_size; }
|
||||||
inline uint64_t get_block_count() { return block_count; }
|
inline uint64_t get_block_count() { return block_count; }
|
||||||
inline uint64_t get_free_block_count() { return data_alloc->get_free_count(); }
|
inline uint64_t get_free_block_count() { return data_alloc->get_free_count(); }
|
||||||
inline uint32_t get_bitmap_granularity() { return disk_alignment; }
|
inline uint32_t get_bitmap_granularity() { return disk_alignment; }
|
||||||
inline uint64_t get_journal_size() { return journal.len; }
|
|
||||||
};
|
};
|
||||||
|
|
|
@ -153,73 +153,22 @@ journal_entry* prefill_single_journal_entry(journal_t & journal, uint16_t type,
|
||||||
return je;
|
return je;
|
||||||
}
|
}
|
||||||
|
|
||||||
void blockstore_impl_t::prepare_journal_sector_write(int cur_sector, blockstore_op_t *op)
|
void prepare_journal_sector_write(journal_t & journal, int cur_sector, io_uring_sqe *sqe, std::function<void(ring_data_t*)> cb)
|
||||||
{
|
{
|
||||||
// Don't submit the same sector twice in the same batch
|
journal.sector_info[cur_sector].dirty = false;
|
||||||
if (!journal.sector_info[cur_sector].submit_id)
|
|
||||||
{
|
|
||||||
io_uring_sqe *sqe = get_sqe();
|
|
||||||
// Caller must ensure availability of an SQE
|
|
||||||
assert(sqe != NULL);
|
|
||||||
ring_data_t *data = ((ring_data_t*)sqe->user_data);
|
|
||||||
journal.sector_info[cur_sector].written = true;
|
journal.sector_info[cur_sector].written = true;
|
||||||
journal.sector_info[cur_sector].submit_id = ++journal.submit_id;
|
|
||||||
journal.submitting_sectors.push_back(cur_sector);
|
|
||||||
journal.sector_info[cur_sector].flush_count++;
|
journal.sector_info[cur_sector].flush_count++;
|
||||||
|
ring_data_t *data = ((ring_data_t*)sqe->user_data);
|
||||||
data->iov = (struct iovec){
|
data->iov = (struct iovec){
|
||||||
(journal.inmemory
|
(journal.inmemory
|
||||||
? journal.buffer + journal.sector_info[cur_sector].offset
|
? journal.buffer + journal.sector_info[cur_sector].offset
|
||||||
: journal.sector_buf + journal.block_size*cur_sector),
|
: journal.sector_buf + journal.block_size*cur_sector),
|
||||||
journal.block_size
|
journal.block_size
|
||||||
};
|
};
|
||||||
data->callback = [this, flush_id = journal.submit_id](ring_data_t *data) { handle_journal_write(data, flush_id); };
|
data->callback = cb;
|
||||||
my_uring_prep_writev(
|
my_uring_prep_writev(
|
||||||
sqe, journal.fd, &data->iov, 1, journal.offset + journal.sector_info[cur_sector].offset
|
sqe, journal.fd, &data->iov, 1, journal.offset + journal.sector_info[cur_sector].offset
|
||||||
);
|
);
|
||||||
}
|
|
||||||
journal.sector_info[cur_sector].dirty = false;
|
|
||||||
// But always remember that this operation has to wait until this exact journal write is finished
|
|
||||||
journal.flushing_ops.insert((pending_journaling_t){
|
|
||||||
.flush_id = journal.sector_info[cur_sector].submit_id,
|
|
||||||
.sector = cur_sector,
|
|
||||||
.op = op,
|
|
||||||
});
|
|
||||||
auto priv = PRIV(op);
|
|
||||||
priv->pending_ops++;
|
|
||||||
if (!priv->min_flushed_journal_sector)
|
|
||||||
priv->min_flushed_journal_sector = 1+cur_sector;
|
|
||||||
priv->max_flushed_journal_sector = 1+cur_sector;
|
|
||||||
}
|
|
||||||
|
|
||||||
void blockstore_impl_t::handle_journal_write(ring_data_t *data, uint64_t flush_id)
|
|
||||||
{
|
|
||||||
live = true;
|
|
||||||
if (data->res != data->iov.iov_len)
|
|
||||||
{
|
|
||||||
// FIXME: our state becomes corrupted after a write error. maybe do something better than just die
|
|
||||||
throw std::runtime_error(
|
|
||||||
"journal write failed ("+std::to_string(data->res)+" != "+std::to_string(data->iov.iov_len)+
|
|
||||||
"). in-memory state is corrupted. AAAAAAAaaaaaaaaa!!!111"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
auto fl_it = journal.flushing_ops.upper_bound((pending_journaling_t){ .flush_id = flush_id });
|
|
||||||
if (fl_it != journal.flushing_ops.end() && fl_it->flush_id == flush_id)
|
|
||||||
{
|
|
||||||
journal.sector_info[fl_it->sector].flush_count--;
|
|
||||||
}
|
|
||||||
while (fl_it != journal.flushing_ops.end() && fl_it->flush_id == flush_id)
|
|
||||||
{
|
|
||||||
auto priv = PRIV(fl_it->op);
|
|
||||||
priv->pending_ops--;
|
|
||||||
assert(priv->pending_ops >= 0);
|
|
||||||
if (priv->pending_ops == 0)
|
|
||||||
{
|
|
||||||
release_journal_sectors(fl_it->op);
|
|
||||||
priv->op_state++;
|
|
||||||
ringloop->wakeup();
|
|
||||||
}
|
|
||||||
journal.flushing_ops.erase(fl_it++);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
journal_t::~journal_t()
|
journal_t::~journal_t()
|
||||||
|
@ -269,19 +218,3 @@ uint64_t journal_t::get_trim_pos()
|
||||||
// Can't trim journal
|
// Can't trim journal
|
||||||
return used_start;
|
return used_start;
|
||||||
}
|
}
|
||||||
|
|
||||||
void journal_t::dump_diagnostics()
|
|
||||||
{
|
|
||||||
auto journal_used_it = used_sectors.lower_bound(used_start);
|
|
||||||
if (journal_used_it == used_sectors.end())
|
|
||||||
{
|
|
||||||
// Journal is cleared to its end, restart from the beginning
|
|
||||||
journal_used_it = used_sectors.begin();
|
|
||||||
}
|
|
||||||
printf(
|
|
||||||
"Journal: used_start=%08lx next_free=%08lx dirty_start=%08lx trim_to=%08lx trim_to_refs=%ld\n",
|
|
||||||
used_start, next_free, dirty_start,
|
|
||||||
journal_used_it == used_sectors.end() ? 0 : journal_used_it->first,
|
|
||||||
journal_used_it == used_sectors.end() ? 0 : journal_used_it->second
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "crc32c.h"
|
#include "crc32c.h"
|
||||||
#include <set>
|
|
||||||
|
|
||||||
#define MIN_JOURNAL_SIZE 4*1024*1024
|
#define MIN_JOURNAL_SIZE 4*1024*1024
|
||||||
#define JOURNAL_MAGIC 0x4A33
|
#define JOURNAL_MAGIC 0x4A33
|
||||||
|
@ -146,21 +145,8 @@ struct journal_sector_info_t
|
||||||
uint64_t flush_count;
|
uint64_t flush_count;
|
||||||
bool written;
|
bool written;
|
||||||
bool dirty;
|
bool dirty;
|
||||||
uint64_t submit_id;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct pending_journaling_t
|
|
||||||
{
|
|
||||||
uint64_t flush_id;
|
|
||||||
int sector;
|
|
||||||
blockstore_op_t *op;
|
|
||||||
};
|
|
||||||
|
|
||||||
inline bool operator < (const pending_journaling_t & a, const pending_journaling_t & b)
|
|
||||||
{
|
|
||||||
return a.flush_id < b.flush_id || a.flush_id == b.flush_id && a.op < b.op;
|
|
||||||
}
|
|
||||||
|
|
||||||
struct journal_t
|
struct journal_t
|
||||||
{
|
{
|
||||||
int fd;
|
int fd;
|
||||||
|
@ -186,9 +172,6 @@ struct journal_t
|
||||||
bool no_same_sector_overwrites = false;
|
bool no_same_sector_overwrites = false;
|
||||||
int cur_sector = 0;
|
int cur_sector = 0;
|
||||||
int in_sector_pos = 0;
|
int in_sector_pos = 0;
|
||||||
std::vector<int> submitting_sectors;
|
|
||||||
std::set<pending_journaling_t> flushing_ops;
|
|
||||||
uint64_t submit_id = 0;
|
|
||||||
|
|
||||||
// Used sector map
|
// Used sector map
|
||||||
// May use ~ 80 MB per 1 GB of used journal space in the worst case
|
// May use ~ 80 MB per 1 GB of used journal space in the worst case
|
||||||
|
@ -197,7 +180,6 @@ struct journal_t
|
||||||
~journal_t();
|
~journal_t();
|
||||||
bool trim();
|
bool trim();
|
||||||
uint64_t get_trim_pos();
|
uint64_t get_trim_pos();
|
||||||
void dump_diagnostics();
|
|
||||||
inline bool entry_fits(int size)
|
inline bool entry_fits(int size)
|
||||||
{
|
{
|
||||||
return !(block_size - in_sector_pos < size ||
|
return !(block_size - in_sector_pos < size ||
|
||||||
|
@ -217,3 +199,5 @@ struct blockstore_journal_check_t
|
||||||
};
|
};
|
||||||
|
|
||||||
journal_entry* prefill_single_journal_entry(journal_t & journal, uint16_t type, uint32_t size);
|
journal_entry* prefill_single_journal_entry(journal_t & journal, uint16_t type, uint32_t size);
|
||||||
|
|
||||||
|
void prepare_journal_sector_write(journal_t & journal, int sector, io_uring_sqe *sqe, std::function<void(ring_data_t*)> cb);
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue