Skip to content
This repository has been archived by the owner on Sep 17, 2020. It is now read-only.

ARGOeu/poem

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

POEM service for ARGO framework

Description

POEM service is a multi-tenant web application used in ARGO monitoring framework that holds list of services, metrics, metric configurations and Nagios probes used within EGI infrastructure. Services and associated metrics are grouped into POEM profiles that instruct monitoring instances what kind of tests to execute for given service. Additionally, it is a register of probes and Nagios metric configurations exposed to Nagios monitoring instances via REST API and with such integration, it helps in speeding up of testing and deployment of probes.

It is based on Django web framework, specifically extension of its admin interface and several Django packages. EGI users are allowed to sign-in through EGI CheckIn federated authentication mechanism. Application is served with Apache web server and all its data is stored in PostgreSQL database.

POEM service is based on Django 2.x framework and following Django packages from PyPI:

  • django-ajax-selects - ease the creation of auto-complete fields
  • django-modelclone - simplify deriving of Profiles and Metric configuration from existing data
  • django-reversion-compare - provides the diff view for Probes and Metric configuration progress
  • django-reversion - provides the basic versioning for Probes and Metric configurations
  • django-tenant-schemas - provides multi tenancy
  • djangorestframework-api-key - provides non-user token generation and protection
  • djangorestframework - provides Token and Session authenticated REST API
  • djangosaml2 - enable SAML2 login feature
  • psycopg2 - PostgreSQL database adapter

Devel instance: https://poem-devel.argo.grnet.gr/

Production instance: https://poem.egi.eu/

More info: http://argoeu.github.io/guides/poem

Installation

POEM web service is meant to be running with Django 2.x driven by Python 3.x on CentOS 7 and served by installation of Apache from Software Collections. Setting up Python virtual environment based on Python 3.x is prerequisite. Once the virtual environment is set up, installation of the service simply narrows down to creating and installing wheel package that will also pull and install all needed dependencies. Beside wheel dependencies, service also needs some packages to be installed from CentOS 7 repositories:

% yum -y install ca-certificates \
		git which \
		xmlsec1 xmlsec1-openssl

Layout of POEM web service files on the filesystem depends on the location of virtual environment. By the default, service assumes it is:

VENV = /home/pyvenv/poem/
File Types Destination
Configuration - General VENV/etc/poem/poem.conf
Configuration - Logging VENV/etc/poem/poem_logging.conf
Configuration - Apache /opt/rh/httpd24/root/etc/httpd/conf.d/
Cron jobs /etc/cron.d/poem-clearsessions, poem-syncvosf
Database handler VENV/bin/poem-db
Sync (VO, Service types) VENV/bin/poem-syncservtype, poem-syncvo
Profiles handlers VENV/bin/poem-exportprofiles, poem-importprofiles
Token set/create VENV/bin/poem-token
Static data served by Apache VENV/usr/share/poem/
Main application code VENV/lib/python3.6/site-packages/Poem/
Log file VENV/var/log/poem

If the default location of virtual environment is inappropriate and needs to be changed, change of it should be reflected by adapting VENV configuration variable in etc/poem/poem.conf, etc/poem/poem_logging.conf, /etc/httpd/conf.d/poem.conf and site-packages/Poem/settings.py.

Setting up virtual environment based on Python 3.6

CentOS 7 operating system delivers Python 3.6 through Software Collections service that is installed with the following instructions:

% yum -y install centos-release-scl
% yum -y install scl-utils
% yum -y install rh-python36 rh-python36-python-pip

Prebuilt mod-wsgi that is linked against Python 3.6 and Apache from Software Collections service can be installed:

% yum -y install rh-python36-mod_wsgi httpd24-httpd httpd24-mod_ssl

Once the Python 3.6 is installed, it needs to be used to create a new virtual environment named poem:

% scl enable rh-python36 'pip install -U pip'
% scl enable rh-python36 'pip3 install virtualenv virtualenvwrapper'
% scl enable rh-python36 'export VIRTUALENVWRAPPER_PYTHON=/opt/rh/rh-python36/root/bin/python3.6; source /opt/rh/rh-python36/root/usr/bin/virtualenvwrapper.sh; export WORKON_HOME=/home/pyvenv; mkdir -p $WORKON_HOME; mkvirtualenv poem'

Notice how the location of virtual environment is controlled with WORKON_HOME variable. Created virtual environment directory will be $WORKON_HOME/poem. It needs to be aligned with VENV variable in service configuration.

Afterward, the context of virtual environment can be started:

% workon poem

As virtual environment is tied to Python 3.6 versions, it's advisable to put helpers/venv_poem.sh into /etc/profile.d as it will configure login shell automatically.

Installing the POEM service

Creation and installation of wheel package is done in the context of virtual environment which needs to be loaded prior.

% workon poem
% (poem) python setup.py sdist bdist_wheel
% (poem) pip3 install dist/*
% (poem) pip3 install -r requirements_ext.txt

wheel package ships cron jobs and Apache configuration and as it is installed in virtual environment, it can not actually layout files outside of it meaning that system-wide files should be placed manually or by configuration management system:

% (poem) cp $VIRTUAL_ENV/etc/cron.d/poem-clearsessions /etc/cron.d/
% (poem) cp $VIRTUAL_ENV/etc/cron.d/poem-syncvosf /etc/cron.d/
% (poem) ln -f -s $VIRTUAL_ENV/etc/httpd/conf.d/poem.conf /opt/rh/httpd24/root/etc/httpd/conf.d/

Next, the correct permission needs to be set on virtual environment directory:

% (poem) chown -R apache:apache $VIRTUAL_ENV

PostgreSQL server setup

POEM is tested and meant to be running on PostgreSQL 10 DBMS. CentOS 7 operating system delivers PostgreSQL 10 through Software Collections service that is installed with the following instructions:

% yum -y install centos-release-scl
% yum -y install scl-utils
% yum -y install rh-postgresql10

Initalization of database:

% scl enable rh-postgresql10 'postgresql-setup --initdb'

In /var/opt/rh/rh-postgresql10/lib/pgsql/data/pg_hba.conf, change default client authentication to password authentication:

# IPv4 local connections:
host    all             all             127.0.0.1/32            md5
# IPv6 local connections:
host    all             all             ::1/128                 md5

Default ident should be replaced by md5.

Start the service:

% systemctl start rh-postgresql10-postgresql

Set password for default DB user postgres:

% su - postgres -c 'scl enable rh-postgresql10 -- psql'
postgres=# \password postgres

Configuration

Configuration is centered around one file VENV/etc/poem/poem.conf that is split into several sections: [DEFAULT], [GENERAL], [DATABASE] and [SECURITY], which are common to all tenants, and tenant-specific sections which have tenant name in the title: [GENERAL_<tenant_name>], [SUPERUSER_<tenant_name>], and [SYNC_<tenant_name>] (<tenant_name> marks the name of the tenant - those section should exist for every tenant).

DEFAULT

[DEFAULT]
VENV = /home/pyvenv/poem/
  • VENV defines the location of virtual environment directory. Value will be interpolated into the other options.

GENERAL

[GENERAL]
Debug = False
TimeZone = Europe/Zagreb
  • Debug serves for the debugging purposes and tells Django to be verbosive and list the stack calls in case of errors
  • Timezone timezone

DATABASE

[DATABASE]
Name = postgres
User = postgres
Password = postgres
Host = localhost
Port = 5432
  • Name is the name of database to use for tables
  • User, Password are credentials that will be used to authenticate to PostgreSQL server
  • Host is hostname of the PostgreSQL server
  • Port where PostgreSQL server is listening for connections

SECURITY

[SECURITY]
AllowedHosts = FQDN1, FQDN2
CAFile = /etc/pki/tls/certs/DigiCertCA.crt
CAPath = /etc/grid-security/certificates/
HostCert = /etc/grid-security/hostcert.pem
HostKey = /etc/grid-security/hostkey.pem
SecretKeyPath = %(VENV)s/etc/poem/secret_key
  • AllowedHosts should have FQDN name of hosts that will be running POEM service. It can be provided as comma separated list of valid FQDNs and it is used as prevention of HTTP Host Header attacks. FQDNs listed here will be matched against request's Host header exactly.
  • CAFile, CAPath are used by sync scripts to authenticate the server certificate
  • HostCert, HostKey are public and private part of client certificate
  • SecretKeyPath is the location of file containing Django SECRET_KEY that is used for cryptographic signing
    • poem-genseckey provided tool can generate unique and unpredictable value and write it into the file
% workon poem
% poem-genseckey -f $VIRTUAL_ENV/etc/poem/secret_key

Part of the REST API is protected by token so for tenants that consume those API methods, token needs to be generated and distributed. Tokens can be generated by the superuser from the Admin UI page or with the provided poem-token tool. Example is creation a token for the client/tenant EGI:

% workon poem
% poem-token -t EGI -s egi

GENERAL_<tenant_name>

[GENERAL_EGI]
Namespace = hr.cro-ngi.TEST
SamlLoginString = Log in using EGI CHECK-IN
SamlServiceName = ARGO POEM EGI-CheckIN
  • Namespace defines the identifier that will be prepended to every Profile
  • SamlLoginString defines the text presented on the SAML2 button of login page
  • SamlServiceName defines service name in SAML2 configuration

SUPERUSER_<tenant_name>

Initial superuser credentials that can be used to sign in to POEM with username and password.

[SUPERUSER_EGI]
Name = test 
Password = test
Email = test@foo.baar

It is important to note that these options should be specified with correct values before trying to create a superuser in database for the given tenant.

SYNC_<tenant_name>

These control options are used by sync scripts that fetch all available services types from GOCDB-like service and Virtual organizations from Operations portal. Additionally, if GOCDB-like service supports only Basic HTTP Authentication, it should be enabled by setting UsePlainHttpAuth and specifying credentials in HttpUser and HttpPass.

[SYNC_EGI]
UsePlainHttpAuth = False 
HttpUser = xxxx
HttpPass = xxxx
ServiceType = https://goc.egi.eu/gocdbpi/private/?method=get_service_types 
VO = http://operations-portal.egi.eu/xml/voIDCard/public/all/true
Services = https://eosc-hub-devel.agora.grnet.gr/api/v2/service-types/

Creating database and starting the service

Prerequisites for creating the empty database are:

  1. PostgreSQL DB server running on Host, listening on Port and accepting User and Password
  2. [SUPERUSER_<tenant_name>] section is set with desired credentials
  3. SECRET_KEY is generated and placed in $VIRTUAL_ENV/etc/poem/secret_key
  4. $VIRTUAL_ENV has permissions set to apache:apache
  5. poem.conf Apache configuration is presented in /opt/rh/httpd24/root/etc/httpd/conf.d/

Once all is set, database can be created with provided tool poem-db. First, a tenant should be created for the public schema in the database, and migrations for public schema are run by calling poem-db -c:

% workon poem
% poem-db -c 
[standard:public] === Running migrate for schema public
[standard:public] Operations to perform:
[standard:public]   Apply all migrations: admin, auth, contenttypes, poem, rest_framework_api_key, reversion, sessions, tenants
[standard:public] Running migrations:
[standard:public]   Applying contenttypes.0001_initial...
[standard:public]  OK
[standard:public]   Applying contenttypes.0002_remove_content_type_name...
[standard:public]  OK
[standard:public]   Applying auth.0001_initial...
[standard:public]  OK
[standard:public]   Applying auth.0002_alter_permission_name_max_length...
[standard:public]  OK
[standard:public]   Applying auth.0003_alter_user_email_max_length...
[standard:public]  OK
[standard:public]   Applying auth.0004_alter_user_username_opts...
[standard:public]  OK
[standard:public]   Applying auth.0005_alter_user_last_login_null...
[standard:public]  OK
[standard:public]   Applying auth.0006_require_contenttypes_0002...
[standard:public]  OK
[standard:public]   Applying auth.0007_alter_validators_add_error_messages...
[standard:public]  OK
[standard:public]   Applying auth.0008_alter_user_username_max_length...
[standard:public]  OK
[standard:public]   Applying auth.0009_alter_user_last_name_max_length...
[standard:public]  OK
[standard:public]   Applying poem.0001_initial...
[standard:public]  OK
[standard:public]   Applying admin.0001_initial...
[standard:public]  OK
[standard:public]   Applying admin.0002_logentry_remove_auto_add...
[standard:public]  OK
[standard:public]   Applying reversion.0001_squashed_0004_auto_20160611_1202...
[standard:public]  OK
[standard:public]   Applying poem.0002_extrev_add...
[standard:public]  OK
[standard:public]   Applying poem.0004_reversion_dbfill...
[standard:public]  OK
[standard:public]   Applying poem.0005_add_metrictype...
[standard:public]  OK
[standard:public]   Applying poem.0006_set_passive...
[standard:public]  OK
[standard:public]   Applying poem.0007_allownull_metricgroup...
[standard:public]  OK
[standard:public]   Applying poem.0008_probe_ondeletenull...
[standard:public]  OK
[standard:public]   Applying poem.0009_poem_nameunique...
[standard:public]  OK
[standard:public]   Applying poem.0010_service...
[standard:public]  OK
[standard:public]   Applying rest_framework_api_key.0001_initial...
[standard:public]  OK
[standard:public]   Applying rest_framework_api_key.0002_auto_20180922_1759...
[standard:public]  OK
[standard:public]   Applying rest_framework_api_key.0003_auto_20180924_1006...
[standard:public]  OK
[standard:public]   Applying rest_framework_api_key.0004_auto_20180924_1303...
[standard:public]  OK
[standard:public]   Applying sessions.0001_initial...
[standard:public]  OK
[standard:public]   Applying tenants.0001_initial...
[standard:public]  OK

Next step is creation of tenants by calling poem-tenant tool. The script takes two mandatory arguments, tenant name and hostname (called domain_url in database). Script will create schema in the database for the given tenant, and the migrations will be run. Also, data is imported for the poem_tags table in the new schema. Superuser is created by calling tool poem-db -u -n <schema_name>:

% poem-tenant -t EGI -d egi-ui.argo.grnet.gr
[standard:egi] === Running migrate for schema egi
[standard:egi] Operations to perform:
[standard:egi]   Apply all migrations: admin, auth, contenttypes, poem, rest_framework_api_key, reversion, sessions, tenants
[standard:egi] Running migrations:
[standard:egi]   Applying contenttypes.0001_initial...
[standard:egi]  OK
[standard:egi]   Applying contenttypes.0002_remove_content_type_name...
[standard:egi]  OK
[standard:egi]   Applying auth.0001_initial...
[standard:egi]  OK
[standard:egi]   Applying auth.0002_alter_permission_name_max_length...
[standard:egi]  OK
[standard:egi]   Applying auth.0003_alter_user_email_max_length...
[standard:egi]  OK
[standard:egi]   Applying auth.0004_alter_user_username_opts...
[standard:egi]  OK
[standard:egi]   Applying auth.0005_alter_user_last_login_null...
[standard:egi]  OK
[standard:egi]   Applying auth.0006_require_contenttypes_0002...
[standard:egi]  OK
[standard:egi]   Applying auth.0007_alter_validators_add_error_messages...
[standard:egi]  OK
[standard:egi]   Applying auth.0008_alter_user_username_max_length...
[standard:egi]  OK
[standard:egi]   Applying auth.0009_alter_user_last_name_max_length...
[standard:egi]  OK
[standard:egi]   Applying poem.0001_initial...
[standard:egi]  OK
[standard:egi]   Applying admin.0001_initial...
[standard:egi]  OK
[standard:egi]   Applying admin.0002_logentry_remove_auto_add...
[standard:egi]  OK
[standard:egi]   Applying reversion.0001_squashed_0004_auto_20160611_1202...
[standard:egi]  OK
[standard:egi]   Applying poem.0002_extrev_add...
[standard:egi]  OK
[standard:egi]   Applying poem.0004_reversion_dbfill...
[standard:egi]  OK
[standard:egi]   Applying poem.0005_add_metrictype...
[standard:egi]  OK
[standard:egi]   Applying poem.0006_set_passive...
[standard:egi]  OK
[standard:egi]   Applying poem.0007_allownull_metricgroup...
[standard:egi]  OK
[standard:egi]   Applying poem.0008_probe_ondeletenull...
[standard:egi]  OK
[standard:egi]   Applying poem.0009_poem_nameunique...
[standard:egi]  OK
[standard:egi]   Applying poem.0010_service...
[standard:egi]  OK
[standard:egi]   Applying rest_framework_api_key.0001_initial...
[standard:egi]  OK
[standard:egi]   Applying rest_framework_api_key.0002_auto_20180922_1759...
[standard:egi]  OK
[standard:egi]   Applying rest_framework_api_key.0003_auto_20180924_1006...
[standard:egi]  OK
[standard:egi]   Applying rest_framework_api_key.0004_auto_20180924_1303...
[standard:egi]  OK
[standard:egi]   Applying sessions.0001_initial...
[standard:egi]  OK
[standard:egi]   Applying tenants.0001_initial...
[standard:egi]  OK
Installed 5 object(s) from 1 fixture(s)
% poem-db -u -n egi
Superuser created successfully.

Afterward, Apache server needs to be started:

% systemctl start httpd24-httpd.service

POEM web application should be now served at https://<tenant_domain_url>/poem

Loading data from .json file

There is also possibility to load data for tenant from .json file:

poem-db -d -n egi -f /path/to/file.json

In case that the data is to be loaded from .json file, super user should not be created beforehand.