Refactoring of code
Changes in structure of the code Changes in the subscription logic Removed requirement for particular package versions Improved connection parameters for InfluxDB Change-Id: Ie61d9c52a4aa66b732c42f9353dd73c34521e991
This commit is contained in:
parent
1d5c54125f
commit
a67c14afc2
3
.hadolint.yaml
Normal file
3
.hadolint.yaml
Normal file
@ -0,0 +1,3 @@
|
||||
ignored:
|
||||
- DL3008
|
||||
- SC1091
|
@ -5,7 +5,7 @@
|
||||
replicaCount: 1
|
||||
|
||||
image:
|
||||
repository: "quay.io/nebulous/monitoring-data-persistor-monitoring-data-persistor"
|
||||
repository: "quay.io/nebulous/monitoring-data-persistor"
|
||||
pullPolicy: IfNotPresent
|
||||
# Overrides the image tag whose default is the chart appVersion.
|
||||
tag: ""
|
||||
|
33
java-spring-boot-demo/.gitignore
vendored
33
java-spring-boot-demo/.gitignore
vendored
@ -1,33 +0,0 @@
|
||||
HELP.md
|
||||
target/
|
||||
!.mvn/wrapper/maven-wrapper.jar
|
||||
!**/src/main/**/target/
|
||||
!**/src/test/**/target/
|
||||
|
||||
### STS ###
|
||||
.apt_generated
|
||||
.classpath
|
||||
.factorypath
|
||||
.project
|
||||
.settings
|
||||
.springBeans
|
||||
.sts4-cache
|
||||
|
||||
### IntelliJ IDEA ###
|
||||
.idea
|
||||
*.iws
|
||||
*.iml
|
||||
*.ipr
|
||||
|
||||
### NetBeans ###
|
||||
/nbproject/private/
|
||||
/nbbuild/
|
||||
/dist/
|
||||
/nbdist/
|
||||
/.nb-gradle/
|
||||
build/
|
||||
!**/src/main/**/build/
|
||||
!**/src/test/**/build/
|
||||
|
||||
### VS Code ###
|
||||
.vscode/
|
@ -1,15 +0,0 @@
|
||||
#
|
||||
# Build stage
|
||||
#
|
||||
FROM docker.io/library/maven:3.9.2-eclipse-temurin-17 AS build
|
||||
COPY src /home/app/src
|
||||
COPY pom.xml /home/app
|
||||
RUN mvn -f /home/app/pom.xml clean package
|
||||
|
||||
#
|
||||
# Package stage
|
||||
#
|
||||
FROM docker.io/library/eclipse-temurin:17-jre
|
||||
COPY --from=build /home/app/target/demo-0.0.1-SNAPSHOT.jar /usr/local/lib/demo.jar
|
||||
EXPOSE 8080
|
||||
ENTRYPOINT ["java","-jar","/usr/local/lib/demo.jar"]
|
@ -1,42 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
<parent>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-parent</artifactId>
|
||||
<version>3.1.0</version>
|
||||
<relativePath/> <!-- lookup parent from repository -->
|
||||
</parent>
|
||||
<groupId>com.example</groupId>
|
||||
<artifactId>demo</artifactId>
|
||||
<version>0.0.1-SNAPSHOT</version>
|
||||
<name>demo</name>
|
||||
<description>Demo project for Spring Boot</description>
|
||||
<properties>
|
||||
<java.version>17</java.version>
|
||||
</properties>
|
||||
<dependencies>
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter</artifactId>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-web</artifactId>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-starter-test</artifactId>
|
||||
<scope>test</scope>
|
||||
</dependency>
|
||||
</dependencies>
|
||||
<build>
|
||||
<plugins>
|
||||
<plugin>
|
||||
<groupId>org.springframework.boot</groupId>
|
||||
<artifactId>spring-boot-maven-plugin</artifactId>
|
||||
</plugin>
|
||||
</plugins>
|
||||
</build>
|
||||
</project>
|
@ -1,13 +0,0 @@
|
||||
package com.example.demo;
|
||||
|
||||
import org.springframework.boot.SpringApplication;
|
||||
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||
|
||||
@SpringBootApplication
|
||||
public class DemoApplication {
|
||||
|
||||
public static void main(String[] args) {
|
||||
SpringApplication.run(DemoApplication.class, args);
|
||||
}
|
||||
|
||||
}
|
@ -1,14 +0,0 @@
|
||||
package com.example.demo;
|
||||
|
||||
import org.springframework.web.bind.annotation.RequestMapping;
|
||||
import org.springframework.web.bind.annotation.RestController;
|
||||
|
||||
@RestController
|
||||
public class DemoController {
|
||||
|
||||
@RequestMapping("/")
|
||||
public Object root() {
|
||||
return null;
|
||||
}
|
||||
|
||||
}
|
@ -1,13 +0,0 @@
|
||||
package com.example.demo;
|
||||
|
||||
import org.junit.jupiter.api.Test;
|
||||
import org.springframework.boot.test.context.SpringBootTest;
|
||||
|
||||
@SpringBootTest
|
||||
class DemoApplicationTests {
|
||||
|
||||
@Test
|
||||
void contextLoads() {
|
||||
}
|
||||
|
||||
}
|
@ -1,42 +1,41 @@
|
||||
# Copyright (c) 2023 Institute of Communication and Computer Systems
|
||||
#
|
||||
# This Source Code Form is subject to the terms of the Mozilla Public
|
||||
# License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||
# file, You can obtain one at https://mozilla.org/MPL/2.0/.
|
||||
# Use python:3.11 as the source stage to build the package
|
||||
FROM python:3.11 as source
|
||||
#RUN pip install --no-cache-dir --upgrade pip
|
||||
|
||||
# Create and set the working directory
|
||||
RUN mkdir /src
|
||||
COPY ./src/ /src/
|
||||
WORKDIR /src
|
||||
|
||||
# Install dependencies and package the application
|
||||
RUN pip install --no-cache-dir -r requirements.txt && python3 setup.py sdist
|
||||
#RUN ls ./dist/
|
||||
|
||||
|
||||
# Start the final stage using Ubuntu as the base image
|
||||
FROM ubuntu:noble
|
||||
|
||||
# Set the environment variables
|
||||
ENV TZ=Europe/Athens
|
||||
ENV VERSION=1.0.0
|
||||
ENV LOG_FILE=monitoring_data_persistor.log
|
||||
ENV LOG_FILE=/home/monitoring_data_persistor.log
|
||||
|
||||
# Set the timezone
|
||||
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone
|
||||
ARG DEBIAN_FRONTEND=noninteractive
|
||||
RUN apt-get update && apt-get install --no-install-recommends -y \
|
||||
libffi8=3.4.4-2 \
|
||||
build-essential=12.10ubuntu1 \
|
||||
libssl-dev=3.0.10-1ubuntu3 \
|
||||
libffi-dev=3.4.4-2 \
|
||||
python3-dev=3.11.4-5ubuntu1 \
|
||||
python3=3.11.4-5ubuntu1 \
|
||||
python3.11-venv=3.11.7-2 \
|
||||
python3-pip=23.3+dfsg-1 \
|
||||
&& rm -rf /var/lib/apt/lists/*
|
||||
|
||||
COPY --from=source ./src/dist/monitoring-data-persistor-$VERSION.tar.gz /home/
|
||||
COPY ./ /home/monitoring-data-persistor-$VERSION
|
||||
# Install the required packages
|
||||
RUN apt-get update && apt-get install --no-install-recommends -y \
|
||||
python3 \
|
||||
python3-venv \
|
||||
python3-pip \
|
||||
&& rm -rf /var/lib/apt/lists/*
|
||||
|
||||
# Copy the packaged application from the source stage
|
||||
COPY --from=source /src/dist/monitoring-data-persistor-$VERSION.tar.gz /home/
|
||||
|
||||
# Set the working directory
|
||||
WORKDIR /home
|
||||
RUN python3 -m venv monitoring-data-env && /bin/sh -c ". monitoring-data-env/bin/activate && pip install --no-cache-dir /home/monitoring-data-persistor-$VERSION.tar.gz"
|
||||
RUN tar -xzvf /home/monitoring-data-persistor-$VERSION.tar.gz && /bin/sh -c ". monitoring-data-env/bin/activate && pip install --no-cache-dir -r /home/monitoring-data-persistor-$VERSION/src/requirements.txt"
|
||||
#RUN bash -x /home/monitoring-data-persistor-$VERSION/src/prepare_project_code.sh
|
||||
#COPY ./ /home/monitoring-data-persistor
|
||||
CMD ["/bin/sh","-c",". monitoring-data-env/bin/activate && python3 -u /home/monitoring-data-persistor-$VERSION/src/main/runtime/DataPersistor.py /home/monitoring-data-persistor-$VERSION/src/resources/config.properties 2>&1 > $LOG_FILE "]
|
||||
|
||||
# Create a virtual environment and install the application package
|
||||
RUN python3 -m venv monitoring-data-env \
|
||||
&& . monitoring-data-env/bin/activate \
|
||||
&& pip install --no-cache-dir monitoring-data-persistor-$VERSION.tar.gz
|
||||
|
||||
CMD ["/bin/bash", "-c", "source monitoring-data-env/bin/activate && python3 -u /home/monitoring-data-persistor-$VERSION/src/main/runtime/DataPersistor.py /home/monitoring-data-persistor-$VERSION/src/resources/config.properties > $LOG_FILE 2>&1"]
|
||||
|
@ -3,8 +3,8 @@ import os
|
||||
|
||||
from proton.reactor import Container
|
||||
|
||||
from main.exn.core import state_publisher, schedule_publisher
|
||||
from main.exn.core.context import Context
|
||||
from exn.core import state_publisher, schedule_publisher
|
||||
from exn.core.context import Context
|
||||
from .core.manager import Manager
|
||||
from .settings import base
|
||||
from .handler import connector_handler
|
@ -1,13 +1,19 @@
|
||||
class Constants:
|
||||
application_name_prefix = "nebulous_"
|
||||
start_forecasting_prefix = "topic://eu.nebulouscloud.forecasting.start_forecasting."
|
||||
forecasting_method_name = "exponentialsmoothing"
|
||||
debug_messages = True
|
||||
metric_list_topic = "metric_list"
|
||||
configuration_file_location = "/home/resources/config.properties"
|
||||
broker_port = "5672"
|
||||
broker_ip = "localhost"
|
||||
broker_username = "admin"
|
||||
broker_password = "admin"
|
||||
monitoring_broker_topic = "monitoring"
|
||||
bucket_name = "nebulous"
|
||||
organization_name = "nebulous"
|
||||
db_token = "tzIfpbU9b77quyvN0yHIbWltSh1c1371-o9nl_wJYaeo5TWdk5txyxXhp2iaLVMvOvf020HnEEAkE0yy5AllKQ=="
|
||||
broker_username = "my-user"
|
||||
broker_password = "my-password"
|
||||
monitoring_broker_topic = "eu.nebulouscloud.monitoring"
|
||||
bucket_name = "my-bucket"
|
||||
organization_name = "my-org"
|
||||
organization_id = "e0033247dcca0c54"
|
||||
db_token = "my-super-secret-auth-token"
|
||||
db_hostname = "localhost"
|
||||
db_port = "8086"
|
||||
monitoring_prefix = "topic://eu.nebulouscloud."+monitoring_broker_topic
|
||||
monitoring_prefix = "topic://"+monitoring_broker_topic+"."
|
@ -8,48 +8,43 @@ from jproperties import Properties
|
||||
from influxdb_client import Point, WritePrecision, InfluxDBClient
|
||||
from influxdb_client.client.write_api import SYNCHRONOUS
|
||||
|
||||
import exn
|
||||
from Constants import Constants
|
||||
from InfluxDBConnector import InfluxDBConnector
|
||||
from main.exn import connector, core
|
||||
from main.exn.handler.connector_handler import ConnectorHandler
|
||||
from exn import connector, core
|
||||
from exn.core.handler import Handler
|
||||
from exn.handler.connector_handler import ConnectorHandler
|
||||
|
||||
logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s - %(message)s')
|
||||
logging.getLogger('main.exn.connector').setLevel(logging.DEBUG)
|
||||
|
||||
|
||||
class Bootstrap(ConnectorHandler):
|
||||
|
||||
pass
|
||||
class ConsumerHandler(Handler):
|
||||
influx_connector = InfluxDBConnector()
|
||||
application_name = ""
|
||||
def __init__(self,application_name):
|
||||
self.application_name = application_name
|
||||
def on_message(self, key, address, body, context, **kwargs):
|
||||
logging.info(f"Received {key} => {address}")
|
||||
application_name = "default_application"
|
||||
if (str(address)).startswith(Constants.monitoring_prefix):
|
||||
if ((str(address)).startswith(Constants.monitoring_prefix) and not (str(address)).endswith(Constants.metric_list_topic)):
|
||||
logging.info("New monitoring data arrived at topic "+address)
|
||||
logging.info(body)
|
||||
point = Point(str(address).split(".")[-1]).field("metricValue",body["metricValue"]).tag("level",body["level"]).tag("component_id",body["component_id"]).tag("application_name",application_name)
|
||||
point = Point(str(address).split(".")[-1]).field("metricValue",body["metricValue"]).tag("level",body["level"]).tag("component_id",body["component_id"]).tag("application_name",self.application_name)
|
||||
point.time(body["timestamp"],write_precision=WritePrecision.S)
|
||||
self.influx_connector.write_data(point)
|
||||
else:
|
||||
print("Address is "+str(address)+", but it was expected for it to start with " + Constants.monitoring_prefix)
|
||||
self.influx_connector.write_data(point,self.application_name)
|
||||
|
||||
class GenericConsumerHandler(Handler):
|
||||
|
||||
def update_properties(configuration_file_location):
|
||||
p = Properties()
|
||||
with open(Constants.configuration_file_location, "rb") as f:
|
||||
p.load(f, "utf-8")
|
||||
Constants.broker_ip, metadata = p["broker_ip"]
|
||||
Constants.broker_port, metadata = p["broker_port"]
|
||||
Constants.broker_username, metadata = p["broker_username"]
|
||||
Constants.broker_password, metadata = p["broker_password"]
|
||||
Constants.monitoring_broker_topic, metadata = p["monitoring_broker_topic"]
|
||||
def on_message(self, key, address, body, context, **kwargs):
|
||||
|
||||
if __name__ == "__main__":
|
||||
Constants.configuration_file_location = sys.argv[1]
|
||||
update_properties(Constants.configuration_file_location)
|
||||
application_handler = Bootstrap()
|
||||
connector = connector.EXN('slovid', handler=application_handler,
|
||||
if (str(address)).startswith(Constants.monitoring_prefix+Constants.metric_list_topic):
|
||||
application_name = body["name"]
|
||||
logging.info("New metrics list message for application "+application_name)
|
||||
connector = exn.connector.EXN('slovid', handler=Bootstrap(),
|
||||
consumers=[
|
||||
core.consumer.Consumer('monitoring', Constants.monitoring_broker_topic + '.>', topic=True, handler=application_handler),
|
||||
core.consumer.Consumer('monitoring', Constants.monitoring_broker_topic + '.>', application=application_name,topic=True, fqdn=True, handler=ConsumerHandler(application_name=application_name)),
|
||||
],
|
||||
url=Constants.broker_ip,
|
||||
port=Constants.broker_port,
|
||||
@ -60,3 +55,35 @@ if __name__ == "__main__":
|
||||
thread = threading.Thread(target=connector.start,args=())
|
||||
thread.start()
|
||||
|
||||
def update_properties(configuration_file_location):
|
||||
p = Properties()
|
||||
with open(configuration_file_location, "rb") as f:
|
||||
p.load(f, "utf-8")
|
||||
Constants.broker_ip, metadata = p["broker_ip"]
|
||||
Constants.broker_port, metadata = p["broker_port"]
|
||||
Constants.broker_username, metadata = p["broker_username"]
|
||||
Constants.broker_password, metadata = p["broker_password"]
|
||||
Constants.monitoring_broker_topic, metadata = p["monitoring_broker_topic"]
|
||||
Constants.organization_name,metadata = p["organization_name"]
|
||||
Constants.bucket_name,metadata = p["bucket_name"]
|
||||
|
||||
if __name__ == "__main__":
|
||||
Constants.configuration_file_location = sys.argv[1]
|
||||
update_properties(Constants.configuration_file_location)
|
||||
component_handler = Bootstrap()
|
||||
|
||||
connector = connector.EXN('slovid', handler=component_handler,
|
||||
consumers=[
|
||||
core.consumer.Consumer('data_persistor_application', Constants.monitoring_broker_topic + '.>', topic=True, fqdn=True, handler=GenericConsumerHandler()),
|
||||
],
|
||||
url=Constants.broker_ip,
|
||||
port=Constants.broker_port,
|
||||
username=Constants.broker_username,
|
||||
password=Constants.broker_password
|
||||
)
|
||||
#connector.start()
|
||||
thread = threading.Thread(target=connector.start,args=())
|
||||
thread.start()
|
||||
|
||||
print("Waiting for messages at the metric list topic, in order to start receiving applications")
|
||||
|
||||
|
@ -1,13 +1,14 @@
|
||||
import logging
|
||||
import threading
|
||||
import time,random
|
||||
import traceback
|
||||
|
||||
from influxdb_client import Point, WritePrecision, InfluxDBClient
|
||||
from influxdb_client.client.write_api import SYNCHRONOUS
|
||||
|
||||
from Constants import Constants
|
||||
from InfluxDBConnector import InfluxDBConnector
|
||||
from main.exn import connector, core
|
||||
from exn import connector, core
|
||||
from datetime import datetime
|
||||
|
||||
|
||||
@ -18,12 +19,12 @@ class Bootstrap(connector.connector_handler.ConnectorHandler):
|
||||
logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s - %(message)s')
|
||||
logging.getLogger('main.exn.connector').setLevel(logging.DEBUG)
|
||||
|
||||
|
||||
application_name = "_Application1"
|
||||
metric_list = ["cpu_usage","ram_usage"]
|
||||
publisher_dict = {}
|
||||
publisher_list = []
|
||||
for metric in metric_list:
|
||||
new_publisher = (core.publisher.Publisher("slovid","monitoring."+metric,topic=True))
|
||||
new_publisher = (core.publisher.Publisher("demopublisher_"+metric,"eu.nebulouscloud.monitoring."+metric,topic=True,fqdn=True))
|
||||
publisher_dict[metric]= new_publisher
|
||||
publisher_list.append(new_publisher)
|
||||
|
||||
@ -55,8 +56,11 @@ for metric_name in metric_list:
|
||||
"component_id":"wordpress_1",
|
||||
"timestamp": time_point
|
||||
}
|
||||
publisher_dict[metric_name].send(body=message)
|
||||
try:
|
||||
publisher_dict[metric_name].send(body=message,application=application_name)
|
||||
if counter%50==0:
|
||||
print("Sending message "+str(counter))
|
||||
counter = counter +1
|
||||
except Exception as e:
|
||||
print(traceback.format_exc())
|
||||
|
||||
|
@ -1,17 +1,53 @@
|
||||
import json
|
||||
|
||||
import requests
|
||||
from influxdb_client import InfluxDBClient, Point, WritePrecision
|
||||
from Constants import Constants
|
||||
from influxdb_client.client.write_api import SYNCHRONOUS
|
||||
|
||||
|
||||
def create_influxdb_bucket(application_name):
|
||||
bucket_name = Constants.application_name_prefix+application_name+"_bucket"
|
||||
|
||||
# Replace with your actual values
|
||||
url = 'http://' + Constants.db_hostname + ':8086/api/v2/buckets'
|
||||
token = Constants.db_token
|
||||
headers = {
|
||||
'Authorization': 'Token {}'.format(token),
|
||||
'Content-Type': 'application/json'
|
||||
}
|
||||
data = {
|
||||
'name': bucket_name,
|
||||
'orgID': Constants.organization_id,
|
||||
'retentionRules': [
|
||||
{
|
||||
'type': 'expire',
|
||||
'everySeconds': 2592000 #30 days (30*24*3600)
|
||||
}
|
||||
]
|
||||
}
|
||||
|
||||
response = requests.post(url, headers=headers, data=json.dumps(data))
|
||||
print(response.status_code)
|
||||
print(response.json())
|
||||
return bucket_name
|
||||
|
||||
|
||||
class InfluxDBConnector:
|
||||
client = InfluxDBClient(url="http://"+Constants.db_hostname+":"+Constants.db_port, token=Constants.db_token, org=Constants.organization_name)
|
||||
write_api = client.write_api(write_options=SYNCHRONOUS)
|
||||
|
||||
def __init__(self):
|
||||
self.client = InfluxDBClient(url="http://"+Constants.db_hostname+":"+Constants.db_port, token=Constants.db_token, org=Constants.organization_name)
|
||||
self.write_api = self.client.write_api(write_options=SYNCHRONOUS)
|
||||
self.influxdb_bucket_created = False
|
||||
self.bucket_name = "demo_bucket"
|
||||
def InfluxDBConnector(self):
|
||||
pass
|
||||
def write_data(self,data):
|
||||
self.write_api.write(bucket=Constants.bucket_name, org=Constants.organization_name, record=data, write_precision=WritePrecision.S)
|
||||
def write_data(self,data,application_name):
|
||||
if not self.influxdb_bucket_created:
|
||||
self.bucket_name = create_influxdb_bucket(application_name)
|
||||
self.influxdb_bucket_created = True
|
||||
|
||||
self.write_api.write(bucket=self.bucket_name, org=Constants.organization_name, record=data, write_precision=WritePrecision.S)
|
||||
|
||||
def get_data(self):
|
||||
query_api = self.client.query_api()
|
||||
|
@ -0,0 +1,19 @@
|
||||
import threading
|
||||
from Constants import Constants
|
||||
from exn import connector, core
|
||||
from main.runtime.DataPersistor import Bootstrap
|
||||
|
||||
if __name__=="__main__":
|
||||
application_handler = Bootstrap()
|
||||
connector = connector.EXN('slovid', handler=application_handler,
|
||||
consumers=[
|
||||
core.consumer.Consumer('monitoring', Constants.monitoring_broker_topic + '.>', topic=True,fqdn=True, handler=application_handler),
|
||||
],
|
||||
url=Constants.broker_ip,
|
||||
port=Constants.broker_port,
|
||||
username=Constants.broker_username,
|
||||
password=Constants.broker_password
|
||||
)
|
||||
#connector.start()
|
||||
thread = threading.Thread(target=connector.start,args=())
|
||||
thread.start()
|
@ -2,4 +2,8 @@ broker_ip=localhost
|
||||
broker_port=5672
|
||||
broker_username=admin
|
||||
broker_password=admin
|
||||
monitoring_broker_topic =monitoring
|
||||
db_username=my-user
|
||||
db_password=my-password
|
||||
monitoring_broker_topic =eu.nebulouscloud.monitoring
|
||||
organization_name=my-org
|
||||
bucket_name=my-bucket
|
@ -4,7 +4,7 @@ setup(
|
||||
name='monitoring-data-persistor',
|
||||
version='1.0.0',
|
||||
#packages=find_packages('.'),
|
||||
packages=["main","main.exn","main.exn.core","main.exn.handler","main.exn.settings","main.runtime"],
|
||||
packages=["main","exn","exn.core","exn.handler","exn.settings","main.runtime"],
|
||||
|
||||
package_dir={'': '.'},
|
||||
entry_points={
|
||||
|
@ -14,9 +14,9 @@
|
||||
container_images:
|
||||
- context: monitoring-data-persistor
|
||||
registry: quay.io
|
||||
repository: quay.io/nebulous/monitoring-data-persistor-monitoring-data-persistor
|
||||
repository: quay.io/nebulous/monitoring-data-persistor
|
||||
namespace: nebulous
|
||||
repo_shortname: monitoring-data-persistor-monitoring-data-persistor
|
||||
repo_shortname: monitoring-data-persistor
|
||||
repo_description: ""
|
||||
|
||||
- job:
|
||||
|
Loading…
Reference in New Issue
Block a user