Add help to surveil-api command
Change-Id: I2c3b94b8d36adb2a87a63f7746be3959e320583e
This commit is contained in:
parent
c3366e70fb
commit
39e5f67b4c
@ -30,12 +30,48 @@ Disabling permissions
|
|||||||
|
|
||||||
Depending on what you are working on, it might be practical to disable permissions. This can be done by editing the ``policy.json`` file found at ``etc/surveil/policy.json``.
|
Depending on what you are working on, it might be practical to disable permissions. This can be done by editing the ``policy.json`` file found at ``etc/surveil/policy.json``.
|
||||||
|
|
||||||
For example, you could modify the following line: ::
|
For example, you could modify the following lines: ::
|
||||||
|
|
||||||
|
"admin_required": "role:admin or is_admin:1",
|
||||||
|
"surveil_required": "role:surveil or rule:admin_required",
|
||||||
|
|
||||||
"surveil:admin": "rule:admin_required",
|
"surveil:admin": "rule:admin_required",
|
||||||
|
"surveil:authenticated": "rule:surveil_required",
|
||||||
|
|
||||||
by: ::
|
by: ::
|
||||||
|
|
||||||
"surveil:admin": "rule:pass",
|
"admin_required": "@",
|
||||||
|
"surveil_required": "@",
|
||||||
|
|
||||||
|
"surveil:admin": "@",
|
||||||
|
"surveil:authenticated": "@",
|
||||||
|
|
||||||
This will modify permissions so that all API calls that require the ``admin`` rule now pass without any verification.
|
This will modify permissions so that all API calls that require the ``admin`` rule now pass without any verification.
|
||||||
|
|
||||||
|
|
||||||
|
Developping the API without docker
|
||||||
|
----------------------------------
|
||||||
|
|
||||||
|
You can get development environment without docker
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
git clone https://review.openstack.org/stackforge/surveil
|
||||||
|
cd surveil
|
||||||
|
virtualenv env
|
||||||
|
source env/bin/activate
|
||||||
|
pip install -r requirements.txt
|
||||||
|
python setup.py develop
|
||||||
|
python setup.py install_data
|
||||||
|
surveil-api -p env/etc/surveil/config.py -a env/etc/surveil/api_paste.ini -c env/etc/surveil/surveil.cfg -r
|
||||||
|
|
||||||
|
Edit your config files
|
||||||
|
|
||||||
|
::
|
||||||
|
|
||||||
|
vim env/etc/surveil/config.py
|
||||||
|
vim env/etc/surveil/surveil.cfg
|
||||||
|
vim env/etc/surveil/policy.json
|
||||||
|
vim env/etc/surveil/api_paste.ini
|
||||||
|
|
||||||
|
Don't forget to start your databases (MongoDB and InfluxDB)
|
||||||
|
@ -12,7 +12,7 @@
|
|||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
from six.moves import configparser
|
import pecan
|
||||||
|
|
||||||
from surveil.api import hooks
|
from surveil.api import hooks
|
||||||
|
|
||||||
@ -22,20 +22,16 @@ server = {
|
|||||||
'host': '0.0.0.0'
|
'host': '0.0.0.0'
|
||||||
}
|
}
|
||||||
|
|
||||||
config = configparser.ConfigParser()
|
authentication = {
|
||||||
config.read("/etc/surveil/surveil.cfg")
|
'config_file': 'policy.json',
|
||||||
|
|
||||||
surveil_api_config = {
|
|
||||||
"mongodb_uri": config.get("surveil", "mongodb_uri"),
|
|
||||||
"ws_arbiter_url": config.get("surveil", "ws_arbiter_url"),
|
|
||||||
"influxdb_uri": config.get("surveil", "influxdb_uri")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
app_hooks = [
|
app_hooks = [
|
||||||
hooks.DBHook(
|
hooks.DBHook(
|
||||||
surveil_api_config['mongodb_uri'],
|
pecan.conf.surveil_api_config['mongodb_uri'],
|
||||||
surveil_api_config['ws_arbiter_url'],
|
pecan.conf.surveil_api_config['ws_arbiter_url'],
|
||||||
surveil_api_config['influxdb_uri']
|
pecan.conf.surveil_api_config['influxdb_uri']
|
||||||
)
|
)
|
||||||
]
|
]
|
||||||
|
|
@ -8,10 +8,12 @@ description-file =
|
|||||||
[files]
|
[files]
|
||||||
packages =
|
packages =
|
||||||
surveil
|
surveil
|
||||||
|
data_files =
|
||||||
|
etc = etc/*
|
||||||
|
|
||||||
[entry_points]
|
[entry_points]
|
||||||
console_scripts =
|
console_scripts =
|
||||||
surveil-api = surveil.cmd.api:main
|
surveil-api = surveil.cmd.api:SurveilCommandRunner.handle_command_line
|
||||||
surveil-init = surveil.cmd.init:main
|
surveil-init = surveil.cmd.init:main
|
||||||
surveil-pack-upload = surveil.cmd.pack_upload:main
|
surveil-pack-upload = surveil.cmd.pack_upload:main
|
||||||
surveil-os-discovery = surveil.cmd.os_discovery:main
|
surveil-os-discovery = surveil.cmd.os_discovery:main
|
||||||
|
@ -12,24 +12,22 @@
|
|||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
|
from __future__ import print_function
|
||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
|
||||||
import os
|
import subprocess
|
||||||
|
import sys
|
||||||
|
import threading
|
||||||
|
import time
|
||||||
|
|
||||||
from paste import deploy
|
from paste import deploy
|
||||||
import pecan
|
import pecan
|
||||||
|
from pecan import commands
|
||||||
|
from pecan import configuration
|
||||||
|
from six.moves import configparser
|
||||||
|
|
||||||
|
|
||||||
def get_config_filename():
|
global_pecan_config_file = None
|
||||||
abspath = os.path.abspath(__file__)
|
|
||||||
path = os.path.dirname(abspath)
|
|
||||||
filename = "config.py"
|
|
||||||
return os.path.join(path, filename)
|
|
||||||
|
|
||||||
|
|
||||||
def get_pecan_config():
|
|
||||||
# Set up the pecan configuration
|
|
||||||
return pecan.configuration.conf_from_file(get_config_filename())
|
|
||||||
|
|
||||||
|
|
||||||
def setup_app(pecan_config):
|
def setup_app(pecan_config):
|
||||||
@ -44,22 +42,116 @@ def setup_app(pecan_config):
|
|||||||
return app
|
return app
|
||||||
|
|
||||||
|
|
||||||
def load_app():
|
|
||||||
return deploy.loadapp('config:/etc/surveil/api_paste.ini')
|
|
||||||
|
|
||||||
|
|
||||||
def app_factory(global_config, **local_conf):
|
def app_factory(global_config, **local_conf):
|
||||||
return VersionSelectorApplication()
|
global global_pecan_config_file
|
||||||
|
return VersionSelectorApplication(global_pecan_config_file)
|
||||||
|
|
||||||
|
|
||||||
class VersionSelectorApplication(object):
|
class VersionSelectorApplication(object):
|
||||||
def __init__(self):
|
def __init__(self, pecan_conf_file):
|
||||||
pc = get_pecan_config()
|
pc = pecan.configuration.conf_from_file(pecan_conf_file)
|
||||||
|
|
||||||
self.v1 = setup_app(pecan_config=pc)
|
self.v1 = setup_app(pecan_config=pc)
|
||||||
self.v2 = setup_app(pecan_config=pc)
|
self.v2 = setup_app(pecan_config=pc)
|
||||||
|
self.config = pc
|
||||||
|
|
||||||
def __call__(self, environ, start_response):
|
def __call__(self, environ, start_response):
|
||||||
if environ['PATH_INFO'].startswith('/v1/'):
|
if environ['PATH_INFO'].startswith('/v1/'):
|
||||||
return self.v1(environ, start_response)
|
return self.v1(environ, start_response)
|
||||||
return self.v2(environ, start_response)
|
return self.v2(environ, start_response)
|
||||||
|
|
||||||
|
|
||||||
|
class SurveilCommand(commands.ServeCommand):
|
||||||
|
|
||||||
|
def run(self, args):
|
||||||
|
global global_pecan_config_file
|
||||||
|
global_pecan_config_file = args.pecan_config
|
||||||
|
super(commands.ServeCommand, self).run(args)
|
||||||
|
app = self.load_app()
|
||||||
|
self.args = args
|
||||||
|
self.serve(app, app.config)
|
||||||
|
|
||||||
|
def load_app(self):
|
||||||
|
config = configparser.ConfigParser()
|
||||||
|
config.read(self.args.config_file)
|
||||||
|
surveil_cfg = {"surveil_api_config":
|
||||||
|
dict([i for i in config.items("surveil")])}
|
||||||
|
configuration.set_config(surveil_cfg, overwrite=True)
|
||||||
|
configuration.set_config(self.args.pecan_config, overwrite=False)
|
||||||
|
|
||||||
|
app = deploy.loadapp('config:%s' % self.args.api_paste_config)
|
||||||
|
app.config = configuration._runtime_conf
|
||||||
|
return app
|
||||||
|
|
||||||
|
def create_subprocess(self):
|
||||||
|
self.server_process = subprocess.Popen(
|
||||||
|
[arg for arg in sys.argv if arg not in ['--reload', '-r']],
|
||||||
|
stdout=sys.stdout, stderr=sys.stderr
|
||||||
|
)
|
||||||
|
|
||||||
|
# TODO(future useless): delete this function when
|
||||||
|
# https://review.openstack.org/#/c/206213/
|
||||||
|
# will be released
|
||||||
|
def watch_and_spawn(self, conf):
|
||||||
|
import watchdog.events as events
|
||||||
|
import watchdog.observers as observers
|
||||||
|
|
||||||
|
print('Monitoring for changes...',
|
||||||
|
file=sys.stderr)
|
||||||
|
|
||||||
|
self.create_subprocess()
|
||||||
|
parent = self
|
||||||
|
|
||||||
|
class AggressiveEventHandler(events.FileSystemEventHandler):
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
self.wait = False
|
||||||
|
|
||||||
|
def should_reload(self, event):
|
||||||
|
for t in (
|
||||||
|
events.FileSystemMovedEvent,
|
||||||
|
events.FileModifiedEvent,
|
||||||
|
events.DirModifiedEvent
|
||||||
|
):
|
||||||
|
if isinstance(event, t):
|
||||||
|
return True
|
||||||
|
return False
|
||||||
|
|
||||||
|
def ignore_events_one_sec(self):
|
||||||
|
if not self.wait:
|
||||||
|
self.wait = True
|
||||||
|
t = threading.Thread(target=self.wait_one_sec)
|
||||||
|
t.start()
|
||||||
|
|
||||||
|
def wait_one_sec(self):
|
||||||
|
time.sleep(1)
|
||||||
|
self.wait = False
|
||||||
|
|
||||||
|
def on_modified(self, event):
|
||||||
|
if self.should_reload(event) and not self.wait:
|
||||||
|
print("Some source files have been modified",
|
||||||
|
file=sys.stderr)
|
||||||
|
print("Restarting server...",
|
||||||
|
file=sys.stderr)
|
||||||
|
parent.server_process.kill()
|
||||||
|
self.ignore_events_one_sec()
|
||||||
|
parent.create_subprocess()
|
||||||
|
|
||||||
|
paths = self.paths_to_monitor(conf)
|
||||||
|
|
||||||
|
event_handler = AggressiveEventHandler()
|
||||||
|
|
||||||
|
for path, recurse in paths:
|
||||||
|
observer = observers.Observer()
|
||||||
|
observer.schedule(
|
||||||
|
event_handler,
|
||||||
|
path=path,
|
||||||
|
recursive=recurse
|
||||||
|
)
|
||||||
|
observer.start()
|
||||||
|
|
||||||
|
try:
|
||||||
|
while True:
|
||||||
|
time.sleep(1)
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
pass
|
||||||
|
@ -17,18 +17,32 @@
|
|||||||
|
|
||||||
"""Access Control Lists (ACL's) control access the API server."""
|
"""Access Control Lists (ACL's) control access the API server."""
|
||||||
|
|
||||||
|
import os
|
||||||
|
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
from oslo_policy import policy
|
from oslo_policy import policy
|
||||||
|
import pecan
|
||||||
|
|
||||||
|
import surveil.api.app as app
|
||||||
|
|
||||||
_ENFORCER = None
|
_ENFORCER = None
|
||||||
|
|
||||||
|
policy_opts = [cfg.StrOpt('project', default='surveil')]
|
||||||
|
|
||||||
policy_opts = [
|
if app.global_pecan_config_file:
|
||||||
cfg.StrOpt('config_dir', default='/etc/surveil/'),
|
pecan_config_dir = os.path.dirname(app.global_pecan_config_file)
|
||||||
cfg.StrOpt('config_file', default='policy.json'),
|
config_dir = cfg.StrOpt('config_dir', default=pecan_config_dir)
|
||||||
cfg.StrOpt('project', default='surveil')
|
else:
|
||||||
]
|
config_dir = cfg.StrOpt('config_dir', default="")
|
||||||
|
if hasattr(pecan.conf, 'authentication'):
|
||||||
|
config_file_name = pecan.conf.authentication.get('config_file',
|
||||||
|
'policy.json')
|
||||||
|
config_file = cfg.StrOpt('config_file', default=config_file_name)
|
||||||
|
else:
|
||||||
|
config_file = cfg.StrOpt('config_file', default='policy.json')
|
||||||
|
|
||||||
|
policy_opts.append(config_dir)
|
||||||
|
policy_opts.append(config_file)
|
||||||
|
|
||||||
CONF = cfg.CONF
|
CONF = cfg.CONF
|
||||||
|
|
||||||
|
@ -16,16 +16,14 @@
|
|||||||
|
|
||||||
from __future__ import print_function
|
from __future__ import print_function
|
||||||
from __future__ import unicode_literals
|
from __future__ import unicode_literals
|
||||||
|
import argparse
|
||||||
import os
|
import os
|
||||||
import subprocess
|
|
||||||
import sys
|
import sys
|
||||||
import threading
|
|
||||||
import time
|
|
||||||
from wsgiref import simple_server
|
|
||||||
|
|
||||||
from oslo_config import cfg
|
from oslo_config import cfg
|
||||||
|
from pecan.commands import base
|
||||||
|
|
||||||
import surveil.api.app as app
|
from surveil.api import app
|
||||||
|
|
||||||
CONF = cfg.CONF
|
CONF = cfg.CONF
|
||||||
|
|
||||||
@ -40,100 +38,50 @@ OPTS = [
|
|||||||
CONF.register_opts(OPTS)
|
CONF.register_opts(OPTS)
|
||||||
|
|
||||||
|
|
||||||
class ServerManager:
|
class SurveilCommandRunner(base.CommandRunner):
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.config = {}
|
super(SurveilCommandRunner, self).__init__()
|
||||||
self.config_file = ""
|
self.parser = argparse.ArgumentParser(description='Surveil API server')
|
||||||
self.server_process = None
|
self.parser.add_argument('--reload', '-r', action='store_true',
|
||||||
self.should_run = True
|
help='Automatically reload as code changes')
|
||||||
|
self.parser.add_argument('--pecan_config', '-p',
|
||||||
|
default='/etc/surveil/config.py',
|
||||||
|
help='Pecan config file (config.py)')
|
||||||
|
self.parser.add_argument('--api_paste_config', '-a',
|
||||||
|
default='/etc/surveil/api_paste.ini',
|
||||||
|
help='API Paste config file (api_paste.ini)')
|
||||||
|
self.parser.add_argument('--config_file', '-c',
|
||||||
|
default='/etc/surveil/surveil.cfg',
|
||||||
|
help='Pecan config file (surveil.cfg)')
|
||||||
|
|
||||||
def run(self, pecan_config, config_file):
|
def run(self, args):
|
||||||
self.config = pecan_config
|
namespace = self.parser.parse_args(args)
|
||||||
self.config_file = config_file
|
# Get absolute paths
|
||||||
|
namespace.pecan_config = os.path.join(os.getcwd(),
|
||||||
|
namespace.pecan_config)
|
||||||
|
namespace.api_paste_config = os.path.join(os.getcwd(),
|
||||||
|
namespace.api_paste_config)
|
||||||
|
namespace.config_file = os.path.join(os.getcwd(),
|
||||||
|
namespace.config_file)
|
||||||
|
|
||||||
if '--reload' in sys.argv:
|
# Check conf files exist
|
||||||
self.watch_and_spawn()
|
if not os.path.isfile(namespace.pecan_config):
|
||||||
else:
|
print("Bad config file: %s" % namespace.pecan_config,
|
||||||
self.start_server()
|
|
||||||
|
|
||||||
def create_subprocess(self):
|
|
||||||
self.server_process = subprocess.Popen(['surveil-api'])
|
|
||||||
|
|
||||||
def start_server(self):
|
|
||||||
pecan_app = app.load_app()
|
|
||||||
host, port = self.config.server.host, self.config.server.port
|
|
||||||
srv = simple_server.make_server(host, port, pecan_app)
|
|
||||||
srv.serve_forever()
|
|
||||||
|
|
||||||
def watch_and_spawn(self):
|
|
||||||
import watchdog.events as events
|
|
||||||
import watchdog.observers as observers
|
|
||||||
|
|
||||||
print('Monitoring for changes...',
|
|
||||||
file=sys.stderr)
|
file=sys.stderr)
|
||||||
|
sys.exit(1)
|
||||||
self.create_subprocess()
|
if not os.path.isfile(namespace.api_paste_config):
|
||||||
parent = self
|
print("Bad config file: %s" % namespace.api_paste_config,
|
||||||
|
|
||||||
class AggressiveEventHandler(events.FileSystemEventHandler):
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
self.wait = False
|
|
||||||
|
|
||||||
def should_reload(self, event):
|
|
||||||
for t in (
|
|
||||||
events.FileSystemMovedEvent,
|
|
||||||
events.FileModifiedEvent,
|
|
||||||
events.DirModifiedEvent
|
|
||||||
):
|
|
||||||
if isinstance(event, t):
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
def ignore_events_one_sec(self):
|
|
||||||
if not self.wait:
|
|
||||||
self.wait = True
|
|
||||||
t = threading.Thread(target=self.wait_one_sec)
|
|
||||||
t.start()
|
|
||||||
|
|
||||||
def wait_one_sec(self):
|
|
||||||
time.sleep(1)
|
|
||||||
self.wait = False
|
|
||||||
|
|
||||||
def on_modified(self, event):
|
|
||||||
if self.should_reload(event) and not self.wait:
|
|
||||||
print("Some source files have been modified",
|
|
||||||
file=sys.stderr)
|
file=sys.stderr)
|
||||||
print("Restarting server...",
|
sys.exit(2)
|
||||||
|
if not os.path.isfile(namespace.config_file):
|
||||||
|
print("Bad config file: %s" % namespace.config_file,
|
||||||
file=sys.stderr)
|
file=sys.stderr)
|
||||||
parent.server_process.kill()
|
sys.exit(1)
|
||||||
self.ignore_events_one_sec()
|
|
||||||
parent.create_subprocess()
|
|
||||||
|
|
||||||
path = self.path_to_monitor()
|
app.SurveilCommand().run(namespace)
|
||||||
|
|
||||||
event_handler = AggressiveEventHandler()
|
@classmethod
|
||||||
|
def handle_command_line(cls): # pragma: nocover
|
||||||
observer = observers.Observer()
|
runner = SurveilCommandRunner()
|
||||||
observer.schedule(
|
runner.run(sys.argv[1:])
|
||||||
event_handler,
|
|
||||||
path=path,
|
|
||||||
recursive=True
|
|
||||||
)
|
|
||||||
observer.start()
|
|
||||||
|
|
||||||
try:
|
|
||||||
while True:
|
|
||||||
time.sleep(1)
|
|
||||||
except KeyboardInterrupt:
|
|
||||||
pass
|
|
||||||
|
|
||||||
def path_to_monitor(self):
|
|
||||||
module = __import__('surveil')
|
|
||||||
return os.path.dirname(module.__file__)
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
srv = ServerManager()
|
|
||||||
srv.run(app.get_pecan_config(), app.get_config_filename())
|
|
||||||
|
@ -18,10 +18,12 @@ import optparse
|
|||||||
import sys
|
import sys
|
||||||
|
|
||||||
import influxdb
|
import influxdb
|
||||||
|
import pecan
|
||||||
|
from pecan import configuration
|
||||||
import pymongo
|
import pymongo
|
||||||
|
from six.moves import configparser
|
||||||
import surveilclient.client as sc
|
import surveilclient.client as sc
|
||||||
|
|
||||||
from surveil.api import config
|
|
||||||
from surveil.cmd import pack_upload
|
from surveil.cmd import pack_upload
|
||||||
|
|
||||||
|
|
||||||
@ -47,18 +49,34 @@ def main():
|
|||||||
dest='packs',
|
dest='packs',
|
||||||
help="Upload/Update configuration packs to MongoDB",
|
help="Upload/Update configuration packs to MongoDB",
|
||||||
action='store_true')
|
action='store_true')
|
||||||
|
parser.add_option('--pecan_config', '-P',
|
||||||
|
default='/etc/surveil/config.py',
|
||||||
|
dest='pecan_config',
|
||||||
|
help='Pecan config file (config.py)')
|
||||||
|
parser.add_option('--config_file', '-c',
|
||||||
|
default='/etc/surveil/surveil.cfg',
|
||||||
|
dest='config_file',
|
||||||
|
help='Pecan config file (surveil.cfg)')
|
||||||
|
|
||||||
opts, _ = parser.parse_args(sys.argv)
|
opts, _ = parser.parse_args(sys.argv)
|
||||||
|
|
||||||
surveil_api_url = 'http://localhost:5311/v2'
|
surveil_api_url = 'http://localhost:5311/v2'
|
||||||
surveil_auth_url = 'http://localhost:5311/v2/auth'
|
surveil_auth_url = 'http://localhost:5311/v2/auth'
|
||||||
surveil_api_version = '2_0'
|
surveil_api_version = '2_0'
|
||||||
|
|
||||||
|
config = configparser.ConfigParser()
|
||||||
|
config.read(opts.config_file)
|
||||||
|
surveil_cfg = {"surveil_api_config":
|
||||||
|
dict([i for i in config.items("surveil")])}
|
||||||
|
configuration.set_config(surveil_cfg, overwrite=True)
|
||||||
|
configuration.set_config(opts.pecan_config, overwrite=False)
|
||||||
|
|
||||||
cli_surveil = sc.Client(surveil_api_url,
|
cli_surveil = sc.Client(surveil_api_url,
|
||||||
auth_url=surveil_auth_url,
|
auth_url=surveil_auth_url,
|
||||||
version=surveil_api_version)
|
version=surveil_api_version)
|
||||||
|
|
||||||
# Create a basic config in mongodb
|
# Create a basic config in mongodb
|
||||||
mongo = pymongo.MongoClient(config.surveil_api_config['mongodb_uri'])
|
mongo = pymongo.MongoClient(pecan.conf.surveil_api_config['mongodb_uri'])
|
||||||
|
|
||||||
if opts.mongodb is True:
|
if opts.mongodb is True:
|
||||||
# Drop the current shinken config
|
# Drop the current shinken config
|
||||||
@ -69,7 +87,7 @@ def main():
|
|||||||
print("Pre-creating InfluxDB database...")
|
print("Pre-creating InfluxDB database...")
|
||||||
# Create the InfluxDB database
|
# Create the InfluxDB database
|
||||||
influx_client = influxdb.InfluxDBClient.from_DSN(
|
influx_client = influxdb.InfluxDBClient.from_DSN(
|
||||||
config.surveil_api_config['influxdb_uri']
|
pecan.conf.surveil_api_config['influxdb_uri']
|
||||||
)
|
)
|
||||||
|
|
||||||
databases = influx_client.get_list_database()
|
databases = influx_client.get_list_database()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user