• Non ci sono risultati.

3.6 Flussi di navigazione

4.1.2 Implementazione dei servizi web

La scelta di PHP come linguaggio di scripting può essere riassunta in aggiunta a quanto detto in precedenza nei seguenti punti principali:

• Ottima integrazione con MySQL

• Disponibilità di API per la conversione di oggetti in formato JSON

• L'esecuzione dei servizi non necessita di un web server, come sarebbe stato necessario per esempio se si avesse scelto Java; infatti i servizi, sono dei file in PHP hostati sul server, che in questo caso risiede nella piattaforma di Altervista.

I servizi sono stati realizzati seguendo la Tabella 1 'Servizi' esposta nel capitolo 3 Progettazione. Si evidenziano alcune parti ritenute significative con relativa spiegazione:

Conversione dei dati e preparazione per l'invio al Client:

<?php

function deliver_response($success, $status_message, $data,

$dateVers){

$response['success']= $success;

$response['status_message']= $status_message;

$response['data']= $data;

//data per il "versionamento dei dati"

$response['dateVers']=$dateVers;

$json_response=json_encode($response, JSON_UNESCAPED_SLASHES);

echo $json_response;

}

?>

La risposta al Client viene preparata utilizzando la function json_encode alla quale si passano i dati da convertire in formato JSON; tra i valori della risposta si invia:

• Un messaggio di stato, ai fini di una maggiore comprensione della risposta • I dati richiesti, che corrispondo al risultato dell'operazione effettuata dal

servizio sul database

• La data di versionamento, utile per la sincronizzazione tra server e client come discusso in precedenza

Generazione del sessionId

Ulteriore attenzione richiede la realizzazione del servizio di registrazione degli utenti; come descritto in precedenza, al termine del login mediante Social si ha un id univoco che identifica l'utente; questo viene salvato nel backend così da poter identificare correttamente l'account BlaBlaFood associato all'utente; occorre precisare che utilizzare questo identificativo per successive comunicazioni tra client e server non è sicuro. A questo proposito si genera un sessionId lato server che il client utilizzerà per identificarsi come si è evidenziato in progettazione. La generazione di questo id deve essere fatta considerando i seguenti aspetti:

• univocità dell'identificatore, per evitare problemi di identificazione tra gli utenti; avere un id non univoco significherebbe che due utenti diversi potrebbero essere considerati identici per il sistema.

• sicurezza nella generazione, essendo l'identificatore dell'utente

Dopo aver effettuato ricerche in merito, si è giunti alla conclusione di non utilizzare la funzione built-in di PHP uniqid(); questo perché la funzione offerta da PHP, come si legge dalla documentazione ufficiale, non garantisce univocità e può presentare problemi a livello di sicurezza. Anche se la probabilità di collisione di due valori generati con uniqid() è molto bassa si è preferito avere garanzia al riguardo così da mostrare piena attenzione ai punti sopra elencati.

Le ricerche effettuate hanno condotto alla scelta della seguente funzione che si riporta brevemente. Con la seguente si riesce a generare un id univoco con un livello di sicurezza accettabile per il dominio in questione

<?php

//generazione di un id univoco

//si specifica che la funzione non è scritta dal sottoscritto //si faccia riferimento per la fonte della funzione a

//stackoverflow.com e alla documentazione ufficiale di php

function random_text( $type = 'alnum', $length = 8 ) { switch ( $type ) { case 'alnum': $pool = '0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; break; case 'alpha': $pool = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ'; break; case 'hexdec': $pool = '0123456789abcdef'; break; case 'numeric': $pool = '0123456789'; break; case 'nozero': $pool = '123456789'; break; case 'distinct': $pool = '2345679ACDEFHJKLMNPRSTUVWXYZ'; break; default:

$pool = (string) $type;

break;

}

$crypto_rand_secure = function ( $min, $max ) {

$range = $max - $min;

// not so random...

if ( $range < 0 ) return $min;

$log = log( $range, 2 );

// length in bytes

$bytes = (int) ( $log / 8 ) + 1;

$bits = (int) $log + 1; // length in bits

// set all lower bits to 1

$filter = (int) ( 1 << $bits ) - 1;

do {

$rnd = hexdec(

bin2hex( openssl_random_pseudo_bytes( $bytes ) ) ); // discard irrelevant bits

$rnd = $rnd & $filter;

} while ( $rnd >= $range );

return $min + $rnd;

$token = "";

$max = strlen( $pool );

for ( $i = 0; $i < $length; $i++ ) {

$token .= $pool[$crypto_rand_secure( 0, $max )];

}

return $token;

}?>

Un esempio di consumo della funzione:

$sessionId = random_text(numeric, 16); //generation of unique and secure id In questo modo si ha un id numerico univoco di 16 cifre, questo è il session id per lo specifico utente.

Selezione della data di versionamento

Per i servizi che ritornano un insieme di valori (es. servizi con metodo GET) si devono effettuare le seguenti macro operazioni:

• selezione dei dati richiesti

• selezione della data di versionamento

Ai fini di una corretta sincronizzazione tra i client e il server occorre prestare attenzione al selezionamento della data di versionamento. Il servizio esegue prima la ricerca della data massima, e successivamente seleziona i dati dal database; seguendo questo ordine si evitano scenari di errata sincronizzazione; infatti se si selezionassero prima i dati e poi la data di versionamento, ogni dato inserito nel database da altri client tra la prima richiesta (riferita ai dati) e la seconda (riferita alla data di versionamento) andrebbe perso.

Gestione degli annunci

Ulteriore attenzione richiede la gestione degli annunci e delle immagini a essi associate; si sono rispettati i seguenti aspetti:

versionamento del dato stesso.

Ogni cambiamento nella tabella immaginipubblicazione (le immagini dell'annuncio), comporta l'aggiornamento della marcatura temporale dell'immagine e dell'annuncio a cui essa fa riferimento.

• L'eventuale cancellazione del dato non può essere effettuata direttamente, come argomentato in progettazione, ma occorre in prima fase effettuare una cancellazione 'logica' dell'annuncio. Successivamente verrà effettuata l'eliminazione effettiva nel rispetto delle politiche di sincronizzazione del sistema. Per una corretta propagazione della cancellazione, occorre eliminare anche le immagini associate all'annuncio; si effettuano quindi due cancellazioni 'logiche': la prima riferita all'annuncio, la seconda sulle eventuali immagini aggiuntive associate ad esso; in entrambe si aggiorna la 'dateVers' così da indicare ai client l'avvenuto cambiamento.

4.2 Client

La progettazione relativa al Client delineata nel capitolo tre, è nella quasi totalità indipendente dalla specifica implementazione in Android; la scelta di Android come piattaforma di sviluppo è sicuramente stata influenzata dal fatto di non possedere un computer con sistema operativo Mac OS; questo non elimina la possibilità di sviluppare in seconda fase anche un'applicazione per dispositivi con sistema operativo iOS come si parlerà nel capitolo 6 'Conclusioni, sviluppi futuri'. Per lo sviluppo si è usato l'IDE ufficiale Android Studio e come linguaggio Java. 4.2.1 Organizzazione progetto

L'applicazione sviluppata in Android è organizzata nei seguenti package principali:

Model, contenente le classi wrapper delle entità principali Product, User, Publication, Image.

Database, contenente la classe PublicationDbOpenHelper e PublicationDBManager.

Service, servizi utili per il funzionamento dell'applicazione, classe Service per operazioni di delete, e aggiornamento della cache locale, classe ServiceCUD per operazioni di modifica, upload di annunci e immagini correlate.

Receiver, contenente i gestori dei risultati dei servizi.Login, contenente LoginActivity e LoginFragment.

Publication, contenente le activity e fragment riferiti all'entità annuncio, ovvero AddOrEditPublicationActivity, PublicationDetailsActivity WorldPublicationFragment (fragment degli annunci locali), MyBoardFragment (fragment degli annunci dell'utente).

Exception, eccezioni lanciabili dall'applicazione come SaveLocalException, o SaveServerException.

Nel package base, oltre a contenere ciò che è sopra elencato, si hanno le activity HomeActivity, TutorialActivity, ProfileActivity e la classe Utilities.

Nel corso dei prossimi paragrafi si tratteranno più dettagliatamente ogni activity, fragment e servizi.

Dopo aver descritto brevemente l'organizzazione, è doveroso procedere evidenziando degli aspetti riguardanti l'architettura.

Documenti correlati