Skip to main content Link Menu Expand (external link) Document Search Copy Copied

Dettagli implementativi sistema di backend

Durante lo sviluppo della parte backend del sistema, come detto nel precedente capitolo, si è cercato di utilizzare il più possibile un approccio a micro-servizi e di raffinare sempre di più la soluzione adottata, per poter sfruttare al meglio i vantaggi che questo approccio offre. Nelle prossime sezioni verranno descritte le scelte implementative legate all’adozione di tale scelta progettuale.

Adapters HTTP e MQTT

All’interno del progetto ogni servizio, espone i propri adapters che possono essere utilizzati dagli altri componenti del sistema per poter comunicare con lui.

Per la componente backend si è scelto di far comunicare tra loro i servizi mediante il protocollo HTTP, mentre la comunicazione tra il micro-servizio GreenhouseCommunication e il sistema di automazione avviene mediante l’utilizzo del protocollo MQTT descritto precedentemente.

Al fine di rendere reattivi i servizi e di gestire entrambi i protocolli di comunicazione, si è scelto di adottare il framework ad eventi e organizzato con architettura event-loop Vert.x.

La logica di business di Vert.x è eseguita all’interno di uno o più componenti chiamati Verticle (vertice). Ogni Verticle viene eseguito in maniera concorrente rispetto agli altri, senza alcuna condivisione di uno stato. Ogni servizio dell’applicazione prevede l’utilizzo di Verticle incaricati di gestire il servizio stesso.

Per quanto riguarda l’inizializzazione, all’avvio di ogni servizio si effettua il setup degli adapters, grazie all’utilizzo della programmazione asincrona e ai concetti di Future e Promise, i quali rendono l’operazione di setup non bloccante.

L’Adapter che è stato più utilizzato, espone il servizio mediante l’utilizzo del protocollo di comunicazione HTTP.

L’avvio dell’Adapter, come si può osservare dal listato 1, può essere suddiviso in due fasi: la prima, consiste nell’avvio dell’HttpServer incaricato della gestione delle richieste in arrivo e delle rispettive risposte in uscita; la seconda, consiste nella definizione delle rotte attraverso le quali i servizi esterni possono richiedere informazioni o far svolgere operazioni al servizio in oggetto.

Le rotte sono state ideate seguendo le linee guida delle API REST, per cui per ogni risorsa manipolata dal servizio sono state definite rotte differenti, ognuna delle quali si occupa di gestire, mediante l’apposito handler, una o più delle operazioni CRUD (Create, Read, Update, Delete) che possono essere effettuate su di essa.

public void start() {
        HttpServer server = vertx.createHttpServer();
        Router router = Router.router(vertx);

        router.route().handler(BodyHandler.create());
        try {
            router.get(BASE_PATH).handler(this::handleGetGreenhouse);
            router.put(BASE_PATH).handler(this::handlePutModality);
            router.get(MODALITY_PATH).handler(this::handleGetModality);
            router.get(PARAM_PATH).handler(this::handleGetParamValues);

        } catch (Exception ex) {
            log("API setup failed - " + ex.toString());
            return;
        }
        server.requestHandler(router).listen(this.port, this.host);
    }

[Listato 1] Esempio di setup di un adapter HTTP

Per quanto riguarda l’Adapter che gestisce la comunicazione MQTT, come si può osservare nel listato 2, prevede anch’esso due fasi per l’avvio: la prima di connessione al broker MQTT, e la seconda di definizione dei topic gestiti dal servizio con i rispettivi handler.

public void setupAdapter(Promise<Void> startPromise) {
        mqttClient = MqttClient.create(this.getVertx());
        mqttClient.connect(port, host, c ->{
            System.out.println("MQTT adapter connected");
            mqttClient.publishHandler(this::handleNewDataReceived)
                    .subscribe(GREENHOUSE_NEWDATA_TOPIC, qos);
        });
        this.getVertx().eventBus().consumer(BRIGHTNESS_OPERATION_TOPIC, this::handleOperationReceived);
        this.getVertx().eventBus().consumer(SOIL_MOISTURE_OPERATION_TOPIC, this::handleOperationReceived);
        this.getVertx().eventBus().consumer(TEMPERATURE_OPERATION_TOPIC, this::handleOperationReceived);
        this.getVertx().eventBus().consumer(AIR_HUMIDITY_OPERATION_TOPIC, this::handleOperationReceived);

[Listato 2] Esempio di setup di un adapter MQTT

Gli adpters di ogni servizio, sono stati progettati per essere il più indipendenti possibili gli uni dagli altri, attraverso l’utilizzo della programmazione asincrona e dell’EventBus messi a disposizione dal framework Vert.x.

Nel caso del servizio GreenhouseCommunication, infatti, le operazioni da eseguire sul micro-controllore vengo ricevute prima mediante l’Adapter HTTP, le quali vengono elaborate del Model incaricato della loro gestione, che si occuperà di inviarle tramite l’EventBus messo a disposizione da Vert.x all’Adapter MQTT ( listato 2), che come previsto, le comunicherà al sistema di automazione tramite il protocollo MQTT.

Web of things e Thing Description

Come detto precedentemente, i micro-servizi Brightness, Humidity, SoilMoisture e Temperature, aderiscono agli standard del WoT . Il componente chiave degli elementi costitutivi di WoT è la descrizione dell’oggetto mediante la sua Thing description (TD).

Una Thing Description definisce un modello informativo di una thing basato sul vocabolario semantico e una serializzazione basata su JSON. Le TD, grazie alla loro flessibilità, promuovono l’interoperabilità fornendo metadati su una thing leggibili (e comprensibili) dagli esseri umani, come titolo, ID, descrizioni, ecc; descrive, inoltre, tutte le azioni, gli eventi e le proprietà disponibili di una thing come tutti i meccanismi di sicurezza disponibili per accedervi.

La Thing Description adottata da noi è stata scritta seguendo lo standard W3C, per cui è caratterizzata da:

  • @context: definisce lo standard in uso;
  • id: un identificativo univoco che viene assegnato alla thing, di base si tratta dell’indirizzo IP;
  • title: è una stringa sommaria e intuitiva che descrive il dispositivo;
  • description: è una stringa intuitiva che descrive il dispositivo e le sue funzioni;
  • properties: è una mappa di oggetti Property che descrivono gli attributi del dispositivo;
  • actions: è una mappa di oggetti Action che descrivono le funzioni che possono essere eseguite su un dispositivo;
  • events: è una mappa di oggetti Event che definiscono i tipi di eventi che possono essere emessi da un dispositivo.

Di seguito viene riportato un esempio di Thing Description, per la precisione quello del servizio Brightness; in ogni caso gli altri risultano similari, con l’unica eccezione che vengono modificate le azioni, gli eventi e le proprietà disponibili.

{
  "@context":"https://webthings.io/schemas/",
  "id":"http://0.0.0.0:8893/brightness",
  "title":"brightness",
  "description":"web connected system to handle the brightness of a greenhouse.",
  "properties":{
    "value":{
      "title":"current value",
      "description":"the level of light registered.",
      "type":"object",
      "properties":{
        "greenhouseId":{"type":"string"},
        "date":{"type":"date"},
        "value":{"type":"float"}
        }
    },
    "links":[{"href":"/brightness/history?id=&limit="}],
    "history":{
      "title":"history values",
      "description":"the history value of the light registered.",
      "type":"list",
      "properties":{
        "greenhouseId":{"type":"string"},
        "date":{"type":"date"},
        "value":{"type":"float"}
      }
    }
  },
  "actions":{
      "fade":{
        "@type":"FadeAction",
        "title":"fade",
        "description":"fade the lamp to a given level.",
        "input":{"type":"string"},
        "links":[{
          "href":"mqtt://broker.mqtt-dashboard.com:1883",
          "op":{
            "level":{
              "type":"integer",
              "minimum":"0",
              "maximum":"255"
              }
            },
          "mqv:topic":"LUMINOSITY"
        }]
      }
  },
  "events":{
    "newData":{
      "title":"new data",
      "type":"string",
      "description":"new luminosity data.",
      "unit":"lux",
      "links":[{"href":"mqtt://broker.mqtt-dashboard.com:1883","mqv:topic":"dataSG"}]
    }
  }
}

[Listato 3] Thing description del servizio brightness

Docker e docker Compose

Al fine di rendere il deployment del sistema di backend più semplice, si è deciso di utilizzare Docker per poter caricare ed eseguire i diversi micro-servizi.

In particolare, si è deciso di inserire ogni micro-servizio all’interno di un apposito container Docker. Un container è una lightweight Virtual Machine, che è in grado di contenere un’applicazione e il suo ambiente di esecuzione, vengono considerati lightweight, in quanto contengono solo lo stretto necessario per poter eseguire l’applicazione e utilizzano le risorse dell’host sul quale vengono istanziati, consentendoci di avere un ambiente di esecuzione isolato per ogni servizio.

Nel nostro caso il container di ogni micro-servizio, come si può vedere nel listato 4, contiene al suo interno il JDK 11, necessario per l’esecuzione del jar e il .jar contenente il codice, compreso di dipendenze, del servizio che rappresenta. Nell’esempio, il servizio è quello relativo alla gestione del parametro della luminosità, per cui una volta nominato l’immagine brightness_service, in modo da poterla poi identificare, viene caricato al suo interno il .jar del servizio e, non appena questo accade, viene lanciata la sua esecuzione.

Eventualmente, se necessario per il corretto funzionamento del servizio, un container può esporre alcune porte, ad esempio, aggiungendo al Dockerfile il comando EXPOSE 1234, il container sarà in grado di esporre all’esterno la porta 1234.

FROM openjdk:11.0.16 AS brightness_service
WORKDIR /
ADD build/libs/brightnessService-0.1.0-all.jar brightnessService-0.1.0-all.jar
CMD java -jar brightnessService-0.1.0-all.jar

[Listato 4] Dockerfile di un micro-servizio

I diversi container sono in grado di comunicare tra loro tramite una rete costruita ad-hoc; per semplicità di gestione la costruzione dei container e della rete è stata realizzata tramite la scrittura di un apposito file docker-compose.yaml.

All’interno del file, come si può vedere nell’estratto del listato 5, ogni servizio viene identificato dallo stesso nome usato per la creazione della propria immagine, poi, viene definito attraverso container_name il nome da utilizzare per effettuare le richieste HTTP al container; dopodiché si passa alla creazione della rete mediante la specificazione delle dipendenze tra container, ad esempio dal listato è possibile notare come i servizi che necessitano di un database dipendano dal servizio mongodb.

Una volta specificate le dipendenze tra i diversi servizi, si procede con la creazione vera e propria del servizio, specificando quale sotto-progetto dovrà essere utilizzato come contesto di esecuzione e il Dockerfile da utilizzare per la creazione della rispettiva immagine. Infine, vengono mappate le porte del container su quelle dell’host, in modo tale da poter effettuare le richieste al servizio.

version: "3.9"
services:
  mongo:
    container_name: mongodb
    image: mongo:latest
    restart: always
    ports:
      - 27017:27017
    volumes:
      - ./docker-entrypoint-initdb.d/mongo-init.js:/docker-entrypoint-initdb.d/mongo-init.js:ro

  brightness_service:
    container_name: brightness
    depends_on:
      - mongo
    build:
      context: brightnessService
      dockerfile: Dockerfile
    ports:
      - 8893:8893

  ...

  client_communication_gateway:
    container_name: client_communication
    build:
      context: clientCommunicationGateway
      dockerfile: Dockerfile
    extra_hosts:
      - host.docker.internal:host-gateway
    ports:
      - 8890:8890
      - 1234:1234
      - 1235:1235

  greenhouse_service:
    container_name: greenhouse
    depends_on:
      - mongo
      - temperature_service
      - brightness_service
      - soil_moisture_service
      - humidity_service
      - operation_service
      - client_communication_gateway
    build:
      context: greenhouseService
      dockerfile: Dockerfile
    ports:
      - 8889:8889

[Listato 5] Estratto del docker-compose utilizzato