2020-04-27 16:11:54 +03:00
---
2020-07-13 19:28:59 +03:00
title: Testing d'unità automatici
2020-04-27 16:11:54 +03:00
layout: default
root: ../..
idx: 8.5
---
2020-07-13 19:28:59 +03:00
## Introduzione <!-- omit in toc -->
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
I testing d'unità sono uno strumento essenziale nell'assicurarsi che il codice sia corretto.
Questo capitolo ti mostrerà come scrivere questi per le mod e i giochi di Minetest usando Busted.
Scrivere i testing d'unità per le funzioni dove vengono chiamate quelle di Minetest è alquanto difficile, ma per fortuna abbiamo già discusso [nel capitolo precedente ](clean_arch.html ) come strutturare il codice in modo da non complicarci la vita.
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
- [Installare Busted ](#installare-busted )
- [Il tuo primo test ](#il-tuo-primo-test )
2020-12-24 21:40:59 +03:00
- [init.lua ](#initlua )
- [api.lua ](#apilua )
- [tests/api_spec.lua ](#testsapispeclua )
2020-07-13 19:28:59 +03:00
- [Simulare: usare funzioni esterne ](#simulare-usare-funzioni-esterne )
- [Controllare commit con Travis ](#controllare-commit-con-travis )
- [Conclusione ](#conclusione )
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
## Installare Busted
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
Prima di tutto, c'è bisogno di installare LuaRocks.
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
* Windows: segui le istruzioni sulla [wiki di LuaRocks ](https://github.com/luarocks/luarocks/wiki/Installation-instructions-for-Windows ).
2020-04-27 16:11:54 +03:00
* Debian/Ubuntu Linux: `sudo apt install luarocks`
2020-07-13 19:28:59 +03:00
Poi, dovresti installare Busted a livello globale:
2020-04-27 16:11:54 +03:00
sudo luarocks install busted
2020-07-13 19:28:59 +03:00
Infine, controlla che sia installato:
2020-04-27 16:11:54 +03:00
busted --version
2020-07-13 19:28:59 +03:00
## Il tuo primo test
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
Busted è il quadro strutturale (o *framework* ) per eccellenza di Lua.
Quello che fa è cercare i file Lua con il nome che termina in `_spec` , eseguendoli poi in un ambiente Lua a sé stante.
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
miamod/
2020-04-27 16:11:54 +03:00
├── init.lua
├── api.lua
2020-07-13 19:28:59 +03:00
└── test
2020-04-27 16:11:54 +03:00
└── api_spec.lua
### init.lua
```lua
2020-07-13 19:28:59 +03:00
miamod = {}
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
dofile(minetest.get_modpath("miamod") .. "/api.lua")
2020-04-27 16:11:54 +03:00
```
### api.lua
```lua
2020-07-13 19:28:59 +03:00
function miamod.somma(x, y)
2020-04-27 16:11:54 +03:00
return x + y
end
```
### tests/api_spec.lua
```lua
2020-07-13 19:28:59 +03:00
-- Cerca le cose necessarie in package.path = "../?.lua;" .. package.path
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
-- Imposta la globale miamod per far sì che l'API possa scriverci sopra
2020-04-27 16:11:54 +03:00
_G.mymod = {} --_
2020-07-13 19:28:59 +03:00
-- Esegue il file api.lua
2020-04-27 16:11:54 +03:00
require("api")
2020-07-13 19:28:59 +03:00
-- Test vari
describe("somma", function()
it("aggiunge", function()
assert.equals(2, miamod.somma(1, 1))
2020-04-27 16:11:54 +03:00
end)
2020-07-13 19:28:59 +03:00
it("supporta valori negativi", function()
assert.equals(0, miamod.somma(-1, 1))
assert.equals(-2, miamod.somma(-1, -1))
2020-04-27 16:11:54 +03:00
end)
end)
```
2020-07-13 19:28:59 +03:00
Puoi ora eseguire i vari test aprendo un terminale nella cartella della mod ed eseguendo `busted .` .
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
È importante che il file dell'API non crei da sé la tabella, in quanto le variabili globali su Busted funzionano diversamente.
Ogni variabile che dovrebbe essere globale su Minetest è invece un file locale su Busted.
Sarebbe stato un modo migliore per Minetest di gestire le cose, ma è ormai troppo tardi per renderlo realtà.
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
Un'altra cosa da notare è che qualsiasi file si stia testando, bisognerebbe evitare che chiami funzioni al di fuori di esso.
Si tende infatti a scrivere i test che controllino un solo file alla volta.
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
## Simulare: usare funzioni esterne
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
Il simulare (*mocking*) è la pratica di sostituire le funzioni dalle quali la parte di codice da testare è dipendente.
Questo può avere due obiettivi: il primo, la funzione potrebbe non essere disponibile nell'area di testing; il secondo, si potrebbero voler catturare le chiamate alla funzione e gli argomenti da essa passati.
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
Se si sono seguiti i consigli nel capitolo delle [Architetture pulite ](clean_arch.html ), si avrà già un file bello pronto da testare, anche se si dovrà comunque simulare le cose non contenute nell'area di testing (per esempio, la vista quando si testa il controllo/API).
Se invece si è deciso di lasciar perdere quella parte, allora le cose sono un po' più complicate in quanto ci sarà da simulare anche la API di Minetest.
2020-04-27 16:11:54 +03:00
```lua
2020-07-13 19:28:59 +03:00
-- come prima, crea una tabella
2020-04-27 16:11:54 +03:00
_G.minetest = {}
2020-07-13 19:28:59 +03:00
-- Definisce la funzione simulata
local chiamate_chat_send_all = {}
2020-04-27 16:11:54 +03:00
function minetest.chat_send_all(name, message)
2020-07-13 19:28:59 +03:00
table.insert(chiamate_chat_send_all, { nome = name, messaggio = message })
2020-04-27 16:11:54 +03:00
end
2020-07-13 19:28:59 +03:00
-- Test
describe("elenca_aree", function()
it("ritorna una riga per ogni area", function()
chiamate_chat_send_all = {} -- resetta la tabella
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
miamod.elenca_aree_chat("singleplayer", "singleplayer")
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
assert.equals(2, #chiamate_chat_send_all )
2020-04-27 16:11:54 +03:00
end)
2020-07-13 19:28:59 +03:00
it("invia al giocatore giusto", function()
chiamate_chat_send_all = {} -- resetta la tabella
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
miamod.elenca_aree_chat("singleplayer", "singleplayer")
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
for _, chiamata in pairs(chiamate_chat_send_all) do --_
assert.equals("singleplayer", chiamata.nome)
2020-04-27 16:11:54 +03:00
end
end)
2020-07-13 19:28:59 +03:00
-- I due test qui in alto in verità sono inutili in quanto
-- questo li esegue entrambi
it("ritorna correttamente", function()
chiamate_chat_send_all = {} -- resetta la tabella
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
miamod.elenca_aree_chat("singleplayer", "singleplayer")
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
local previsto = {
{ nome = "singleplayer", messaggio = "Town Hall (2,43,63)" },
{ nome = "singleplayer", messaggio = "Airport (43,45,63)" },
2020-04-27 16:11:54 +03:00
}
2020-07-13 19:28:59 +03:00
assert.same(previsto, chiamate_chat_send_all)
2020-04-27 16:11:54 +03:00
end)
end)
```
2020-07-13 19:28:59 +03:00
## Controllare commit con Travis
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
Lo script di Travis usato nel capitolo [Controllo automatico degli errori ](luacheck.html ) può essere modificato per eseguire (anche) Busted
2020-04-27 16:11:54 +03:00
```yml
language: generic
sudo: false
addons:
apt:
packages:
- luarocks
before_install:
- luarocks install --local luacheck && luarocks install --local busted
script:
- $HOME/.luarocks/bin/luacheck .
- $HOME/.luarocks/bin/busted .
notifications:
email: false
```
2020-07-13 19:28:59 +03:00
## Conclusione
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
I testing d'unità aumenteranno notevolmente la qualità e l'affidabilità di un progetto se usati adeguatamente, ma ti richiederanno di strutturare il codice in maniera diversa dal solito.
2020-04-27 16:11:54 +03:00
2020-07-13 19:28:59 +03:00
Per un esempio di mod con molti testing d'unità, vedere la [crafting di rubenwardy ](https://github.com/rubenwardy/crafting ).