This is the 5GTANGO Gatekeeper utils gem, which implements some utilitary features shared across the different Gatekeeper repositories: tng-gtk-common
, tng-gtk-sp
and tng-gtk-vnv
.
The code for this utility library was extracted from those repositories and made available to them so that code shouldn't hang around duplicated.
For details on the overall 5GTANGO architecture please check here and also in the previously mentioned repositories.
This library implements some of the features that were being duplicated in the 5GTANGO Gatekeeper repositories mentioned above, tng-gtk-common
, tng-gtk-sp
and tng-gtk-vnv
.
Ruby's standard library already provides a Logger, which we want to extend and make it providing outputs such as the following (in JSON format -- see this user story, authentication is needed):
{
"type": "I",
"timestamp": "2018-10-18 15:49:08 UTC",
"start_stop": "STOP",
"component": "tng-api-gtw",
"operation": "package upload",
"message": "package uploaded 201",
"status": "201",
"time_elapsed": "00:01:20"
}
These fields have the following meanings/values:
type
: mandatory, can beI
(nfo),W
(arning),D
(ebug),E
(rror),F
(atal) orU
(nknown);timestamp
: mandatory, it's the timestamp of the logged message, in UTC format;start_stop
: optional (can be empty),START
orSTOP
component
: mandatory, holds the name of the component logging the message;operation
: mandatory, holds the operation logging the message;message
: mandatory, the message to be logged;status
: optional, makes sense forstart_stop='STOP'
, e.g., to hold the HTTP status;time_elapsed
: optional, makes sense forstart_stop='STOP'
, to hold the time the operation took to complete.
The usual approach of redefining the formatter in the Logger
standard library (see below) doesn't have the full flexibility we need:
Logger.new(logdev, formatter: proc {|severity, datetime, progname, msg|
JSON.dump(timestamp: "#{datetime.to_s}", message: msg)
})
It should also support a LOGLEVEL
variable that may assume one of the usual values debug
, info
, warning
, error
, fatal
or unknown
(defaults to info
, so only logging messages marked as unknown
, fatal
, error
, warning
or info
are shown -- see this user story).
This output is achieved by calling methods such as Logger.debug(...)
, with the minimum number of mandatory fields.
An example of using the above described logger is the following:
LOG_COMPONENT=self.name
LOGGER=Tng::Gtk::Utils::Logger
...
LOGGER.error(component: LOG_COMPONENT, operation:__method__.to_s, message:"key :uuid is missing in record #{record}")
For example, if the time_elapsed
field is to be filled, we can do the following:
began_at = Time.now.utc
do_some_lengthy_processing()
LOGGER.info(component: LOG_COMPONENT, operation:__method__.to_s, message:"Done!", time_elapsed: Time.now.utc-began_at)
The Tng::Gtk::Utils::Logger
class depends on the following ruby gems:
json
;
The first ...
The Tng::Gtk::Utils
' cache API is detailled next.
To use this, you need to require this library (tng/gtk/utils
):
require `tng/gtk/utils'
Then, in the module
or class
you want to use cache, just extend the module:
extend Tng::Gtk::Utils
Whenever you want to cache an Hash
, just use the cache
macro (or module method
):
cache {uuid: '4345444a-d659-4843-a618-ea43b8a1f9ba', whatever: 'else'}.to_json
For checking if we've got some UUID
cached, just use the cached?
macro:
x = cached? '4345444a-d659-4843-a618-ea43b8a1f9ba'
require 'json'
require `tng/gtk/utils'
class Fetch
extend Tng::Gtk::Utils
class << self
def call(params)
do_validation_stuff
cached = cached? params[:uuid] # now check if we've got this cached
return JSON.parse(cached, symbolize_names: :true) if cached
real_value = fetch_real_value # if here, then it's not cached: fetch real value
cache real_value.to_json # and cache it for next time (and return it)
end
end
end
uses Logger
The Fetch
class works very much like Euby-on-Rails' ActiveModel
gem, without all the complexities around it.
uses Cache and Logger
This gem is implemented in ruby, version 2.4.3.
To install this gem in your system, please do:
git clone https://github.com/sonata-nfv/tng-gtk-utils.git # Clone this repository
cd tng-gtk-utils && gem build tng-gtk-utils.gemspec # Build the gem
[sudo] gem install tng-gtk-utils-0.0.1.gem # Install it (the filename/version may vary)
To install this gem using a Gemfile, please add the following line to that file:
gem 'tng-gtk-utils', git: 'https://github.com/sonata-nfv/tng-gtk-utils'
This section covers all the needs a developer has in order to be able to contribute to this project.
We are using the following libraries (also referenced in the Gemfile
file) for development:
activerecord
(5.2
), the Object-Relational Mapper (ORM);bunny
(2.8.0
), the adapter to the RabbitMQ message queue server;pg
(0.21.0
), the adapter to the PostgreSQL database;puma
(3.11.0
), an application server;rack
(2.0.4
), a web-server interfacing library, on top of whichsinatra
has been built;rake
(12.3.0
), a dependencies management tool for ruby, similar to make;sinatra
(2.0.2
), a web framework for implementing efficient ruby APIs;sinatra-activerecord
(2.0.13
),sinatra-contrib
(2.0.2
), several add-ons tosinatra
;sinatra-cross_origin
(0.4.0
), a middleware tosinatra
that helps in managing theCross Origin Resource Sharing (CORS)
problem;sinatra-logger
(0.3.2
), a logger middleware;
The following gems (libraries) are used just for tests:
ci_reporter_rspec
(1.0.0
), a library for helping in generating continuous integration (CI) test reports;rack-test
(0.8.2
), a helper testing framework forrack
-based applications;rspec
(3.7.0
), a testing framework for ruby;rubocop
(0.52.0
), a library for white box tests;rubocop-checkstyle_formatter
(0.4.0
), a helper library forrubocop
;webmock
(3.1.1
), which alows mocking (i.e., faking) HTTP calls;
These libraries are installed/updated in the developer's machine when running the command (see above):
$ bundle install
We usually use rbenv
as the ruby version manager, but others like rvm
may work as well.
Developing this library is straightforward with a low amount of necessary steps.
Changes to the repository can be requested using this repository's issues and pull requests mechanisms.
The most up-to-date version is v4. For the versions available, see the link to tags on this repository.
The configuration of the micro-service is done through the following environment variables, defined in the Dockerfile:
CATALOGUE_URL
, which defines the Catalogue's URL, where test descriptors are fetched from;REPOSITORY_URL
, which defines the Repository's URL, where test plans and test results are fetched from;DATABASE_URL
, which defines the database's URL, in the following format:postgresql://user:password@host:port/database_name
(Note: this is an alternative format to the one described in the Installing from the Docker container section);MQSERVER_URL
, which defines the message queue server's URL, in the following format:amqp://user:password@host:port
Unit tests are defined for both controllers
and services
, in the /spec
folder. Since we use rspec
as the test library, we configure tests in the spec_helper.rb
file, also in the /spec
folder.
These tests are executed by running the following command:
$ bundle exec rspec spec
Wider scope (integration and functional) tests involving this micro-service are defined in tng-tests
.
Our style guide is really simple:
- We try to follow a Clean Code philosophy in as much as possible, i.e., classes and methods should do one thing only, have the least number of parameters possible, etc.;
- we use two spaces for identation.
We have specified this micro-service's API in a swagger-formated file. Please check it there.
This 5GTANGO component is published under Apache 2.0 license. Please see the LICENSE file for more details.
- Please use the GitHub issues to report bugs.
- You may use the mailing list [email protected]