I test di unità ti consentono di controllare la qualità del codice dopo averlo scritto, ma puoi anche utilizzarli per migliorare il processo di sviluppo man mano che procedi. Anziché scrivere i test dopo aver completato lo sviluppo dell'applicazione, ti consigliamo di scriverli man mano. In questo modo, puoi progettare unità di codice piccole, manutenibili e riutilizzabili. Inoltre, ti consente di testare il codice in modo accurato e rapido.
Quando esegui test di unità locale, esegui test che rimangono all'interno del tuo ambiente di sviluppo senza coinvolgere componenti remoti. App Engine fornisce utilità di test che utilizzano implementazioni locali di Datastore e di altri servizi App Engine. Ciò significa che puoi verificare l'utilizzo di questi servizi da parte del tuo codice localmente, senza eseguire il deployment del codice in App Engine, utilizzando gli stub di servizio.
Uno stub di servizio è un metodo che simula il comportamento del servizio. Ad esempio, lo stub del servizio Datastore mostrato in Scrivere test di Datastore e Memcache ti consente di testare il codice di Datastore senza effettuare richieste al Datastore reale. Qualsiasi entità archiviata durante un test di unità del datastore viene memorizzata in memoria, non nel datastore, ed eliminata al termine dell'esecuzione del test. Puoi eseguire test piccoli e rapidi senza alcuna dipendenza dal datastore stesso.
Questo documento descrive come scrivere test di unità per diversi servizi App Engine locali e fornisce alcune informazioni sulla configurazione di un framework di test.
Introduzione agli strumenti di test di Python 2
Un modulo Python di App Engine chiamato
testbed
rende disponibili gli stub di servizio per i test di unità.
Gli stub di servizio sono disponibili per i seguenti servizi:
- App Identity
init_app_identity_stub
- Blobstore (utilizza
init_blobstore_stub
) - Capacità (utilizza
init_capability_stub
) - Datastore (utilizza
init_datastore_v3_stub
) - File (utilizza
init_files_stub
) - Immagini (solo per
dev_appserver;
utilizza
init_images_stub
) - LogService (utilizza
init_logservice_stub
) - Posta (utilizza
init_mail_stub
) - Memcache (utilizza
init_memcache_stub
) - Coda di attività (utilizza
init_taskqueue_stub
) - Recupero URL (utilizza
init_urlfetch_stub
) - Servizio utente (utilizza
init_user_stub
)
Per inizializzare tutti gli stub contemporaneamente, puoi utilizzare init_all_stubs
.
Scrittura di test di Datastore e memcache
Questa sezione mostra un esempio di come scrivere codice che testa l'utilizzo dei servizi datastore e memcache.
Assicurati che il tuo programma di test abbia le librerie appropriate nel percorso di caricamento di Python, incluse le librerie di App Engine, yaml
(incluse nell'SDK di App Engine), la radice dell'applicazione e eventuali altre modifiche al percorso della libreria previste dal codice dell'applicazione (ad esempio una directory ./lib
locale, se ne hai una). Ad esempio:
import sys
sys.path.insert(1, 'google-cloud-sdk/platform/google_appengine')
sys.path.insert(1, 'google-cloud-sdk/platform/google_appengine/lib/yaml/lib')
sys.path.insert(1, 'myapp/lib')
Importa il modulo unittest
di Python e i moduli App Engine pertinenti ai servizi in fase di test, in questo caso memcache
e ndb
, che utilizzano sia Datastore sia Memcache. Importa anche il modulo testbed
.
Poi crea un corso TestModel
. In questo esempio, una funzione controlla se un'entità è archiviata in memcache. Se non viene trovata alcuna entità, ne viene cercata un'altra nel datastore. Spesso questo può essere ridondante nella vita reale, poiché ndb
utilizza memcache stesso dietro le quinte, ma è comunque un pattern accettabile per un
test.
A questo punto, crea un test case. Indipendentemente dai servizi che stai testando, il caso di test deve creare un'istanza Testbed
e attivarla. Il caso di test deve anche inizializzare gli stub di servizio pertinenti, in questo caso utilizzando init_datastore_v3_stub
e init_memcache_stub
. I metodi per inizializzare
altri stub di servizi App Engine sono elencati in Introduzione alle utilità di test di Python.
Il metodo init_datastore_v3_stub()
senza argomenti utilizza un datastore in memoria inizialmente vuoto. Se vuoi testare un'entità
dell'area dati esistente, includi il relativo percorso come argomento di init_datastore_v3_stub()
.
Oltre a setUp()
, includi un metodo tearDown()
che disattivi il
laboratorio. In questo modo vengono ripristinati gli stub originali in modo che i test non interferiscano tra loro.
Poi implementa i test.
Ora puoi utilizzare TestModel
per scrivere test che utilizzano gli stub dei servizi datastore o memcache anziché i servizi reali.
Ad esempio, il metodo mostrato di seguito crea due entità: la prima utilizza il valore predefinito per l'attributo number
(42) e la seconda utilizza un valore non predefinito per number
(17). Il metodo genera quindi una query per le entità TestModel
, ma solo per quelle con il valore predefinito number
.
Dopo aver recuperato tutte le entità corrispondenti, il metodo verifica che sia stata trovata esattamente un'entità e che il valore dell'attributo number
di quell'entità sia il valore predefinito.
Come altro esempio, il seguente metodo crea un'entità e la recupera utilizzando la funzione GetEntityViaMemcache()
che abbiamo creato sopra. Il metodo
quindi verifica che sia stata restituita un'entità e che il suo valore number
sia lo stesso
dell'entità creata in precedenza.
Infine, invoca unittest.main()
.
Per eseguire i test, consulta Eseguire i test.
Scrittura di test di Cloud Datastore
Se la tua app utilizza Cloud Datastore, ti consigliamo di scrivere test che verifichino il comportamento dell'applicazione in caso di eventuale coerenza.
db.testbed
offre opzioni che semplificano questa procedura:
La classe PseudoRandomHRConsistencyPolicy
ti consente di controllare la probabilità che un'operazione di scrittura venga applicata prima di ogni query globale (non principale). Impostando la probabilità su 0%, diciamo allo stub del datastore di operare con la massima coerenza finale. La coerenza finale massima significa che le scritture verranno committate, ma non verranno mai applicate, pertanto le query globali (non antecedenti) non riusciranno mai a vedere le modifiche. Ovviamente, questo non è rappresentativo della quantità di coerenza finale che la tua applicazione vedrà durante l'esecuzione in produzione, ma a scopo di test è molto utile poter configurare il datastore locale in modo che si comporti sempre in questo modo. Se utilizzi una probabilità diversa da zero, PseudoRandomHRConsistencyPolicy
genera una sequenza deterministica di decisioni di coerenza in modo che i risultati del test siano coerenti:
Le API di test sono utili per verificare che l'applicazione si comporti correttamente in presenza di coerenza finale, ma tieni presente che il modello locale di coerenza di lettura con replica elevata è un'approssimazione del modello di coerenza di lettura con replica elevata di produzione, non una replica esatta. Nell'ambiente local, l'esecuzione di un get()
di un Entity
che appartiene a un gruppo di entità con una scrittura non applicata rende sempre visibili i risultati della scrittura non applicata alle query globali successive. In produzione non è così.
Scrivere test di posta
Puoi utilizzare lo stub del servizio di posta per testare il servizio mail. Come per altri servizi supportati dal testbed, all'inizio devi inizializzare lo stub, poi invocare il codice che utilizza l'API Mail e infine verificare se sono stati inviati i messaggi corretti.
Scrivere test delle coda di attività
Puoi utilizzare lo stub taskqueue per scrivere test che utilizzano il servizio taskqueue. Come per gli altri servizi supportati dal testbed, all'inizio devi inizializzare lo stub, poi invocare il codice che utilizza l'API taskqueue e infine verificare se le attività sono state aggiunte correttamente alla coda.
Impostazione del file di configurazione queue.yaml
Se vuoi eseguire test sul codice che interagisce con una coda non predefinita, devi creare e specificare un file queue.yaml
da utilizzare per l'applicazione.
Di seguito è riportato un esempio di queue.yaml
:
Per ulteriori informazioni sulle opzioni disponibili per queue.yaml, consulta la configurazione della coda di attività.
La posizione di queue.yaml
viene specificata durante l'inizializzazione dello stub:
self.testbed.init_taskqueue_stub(root_path='.')
Nel sample, queue.yaml
si trova nella stessa directory dei test. Se fosse in un'altra cartella, il percorso dovrebbe essere specificato in root_path
.
Filtrare le attività
get_filtered_tasks
dello stub taskqueue consente di filtrare le attività in coda.
In questo modo è più facile scrivere test che devono verificare il codice che mette in coda più attività.
Scrivere test delle attività differite
Se il codice dell'applicazione utilizza la libreria differita, puoi utilizzare lo stub taskqueue insieme a deferred
per verificare che le funzioni differite vengano messe in coda ed eseguite correttamente.
Modificare le variabili di ambiente predefinite
I servizi App Engine spesso dipendono dalle variabili di ambiente. Il metodo activate()
della classe testbed.Testbed
utilizza valori predefiniti per questi, ma puoi impostare valori personalizzati in base alle tue esigenze di test con il metodo setup_env
della classe testbed.Testbed
.
Ad esempio, supponiamo che tu abbia un test che memorizza diverse entità nel datastore, tutte collegate allo stesso ID applicazione. Ora vuoi eseguire nuovamente gli stessi test, ma utilizzando un ID applicazione diverso da quello collegato alle entità memorizzate. Per farlo, passa il nuovo valore a self.setup_env()
come app_id
.
Ad esempio:
Simulazione di accesso
Un altro utilizzo frequente di setup_env
è simulare l'accesso di un utente, con o senza privilegi amministrativi, per verificare se i gestori funzionano correttamente in ogni caso.
Ora i metodi di test possono chiamare, ad esempio, self.loginUser('', '')
per simulare l'assenza di un utente che ha eseguito l'accesso, self.loginUser('test@example.com', '123')
per simulare l'accesso di un utente non amministratore e self.loginUser('test@example.com',
'123', is_admin=True)
per simulare l'accesso di un utente amministratore.
Configurazione di un framework di test
Le utilità di test dell'SDK non sono legate a un framework specifico. Puoi eseguire i test di unità con qualsiasi test runner App Engine disponibile, ad esempio nose-gagg o ferrisnose. Puoi anche scrivere un semplice test runner o utilizzare quello mostrato di seguito.
I seguenti script utilizzano il modulo unittest di Python.
Puoi assegnare allo script il nome che preferisci. Quando lo esegui, fornisci il percorso dell'installazione di Google Cloud CLI o Google App Engine SDK e il percorso dei moduli di test. Lo script rileverà tutti i test nel percorso specificato e stamperà i risultati nello stream di errori standard. I file di test seguono la convenzione di avere il prefisso test
al nome.
Esecuzione dei test
Puoi eseguire questi test semplicemente eseguendo lo script runner.py
, descritto in dettaglio nella sezione Configurare un framework di test:
python runner.py <path-to-appengine-or-gcloud-SDK> .