minetest_modding_book/_en/quality/common_mistakes.md

167 lines
4.7 KiB
Markdown
Raw Normal View History

2018-04-03 15:11:27 +03:00
---
title: Common Mistakes
layout: default
2018-07-15 21:36:35 +03:00
root: ../..
2018-09-14 17:44:16 +03:00
idx: 7.1
2018-07-15 21:13:16 +03:00
redirect_from: /en/chapters/common_mistakes.html
2018-04-03 15:11:27 +03:00
---
## Introduction
This chapter will detail common mistakes that are made, and how to avoid them.
* [Never Store ObjectRefs (ie: players or entities)](#never-store-objectrefs-ie-players-or-entities)
* [Don't Trust Formspec Submissions](#dont-trust-formspec-submissions)
* [Set ItemStacks After Changing Them](#set-itemstacks-after-changing-them)
## Never Store ObjectRefs (ie: players or entities)
If the Object a ObjectRef represents is deleted - for example, the player goes
offline or the entity is unloaded - then calling any methods will result in a crash.
Don't do this:
```lua
2018-04-03 15:11:27 +03:00
minetest.register_on_joinplayer(function(player)
local function func()
local pos = player:get_pos() -- BAD!
-- `player` is stored in the scope then accessed later.
-- If the player leaves in that second, the server *will* crash.
end
minetest.after(1, func)
foobar[player:get_player_name()] = player
-- RISKY
-- It's recommended to just not do this
2018-09-24 19:16:00 +03:00
-- use minetest.get_connected_players() and
-- minetest.get_player_by_name() instead.
2018-04-03 15:11:27 +03:00
end)
```
2018-04-03 15:11:27 +03:00
instead, do this:
```lua
2018-04-03 15:11:27 +03:00
minetest.register_on_joinplayer(function(player)
local function func(name)
-- Attempt to get the ref again
local player = minetest.get_player_by_name(name)
-- Check that the player is still online
if player then
-- Yay! This is fine
local pos = player:get_pos()
end
end
-- Pass the name into the function
minetest.after(1, func, player:get_player_name())
end)
```
2018-04-03 15:11:27 +03:00
## Don't Trust Formspec Submissions
Malicious clients can submit formspecs whenever they like with whatever content
they like.
The following code has a vulnerability where any player can give
themselves moderator privileges:
2018-04-03 15:11:27 +03:00
```lua
2018-04-03 15:11:27 +03:00
local function show_formspec(name)
if not minetest.check_player_privs(name, { privs = true }) then
return false
end
minetest.show_formspec(name, "modman:modman", [[
size[3,2]
field[0,0;3,1;target;Name;]
button_exit[0,1;3,1;sub;Promote]
]])
return true
})
2018-09-24 19:16:00 +03:00
minetest.register_on_player_receive_fields(function(player,
formname, fields)
2018-04-03 15:11:27 +03:00
-- BAD! Missing privilege check here!
local privs = minetest.get_player_privs(fields.target)
privs.kick = true
privs.ban = true
minetest.set_player_privs(fields.target, privs)
return true
end)
```
2018-04-03 15:11:27 +03:00
Instead, do this:
```lua
2018-09-24 19:16:00 +03:00
minetest.register_on_player_receive_fields(function(player,
formname, fields)
2018-04-03 15:11:27 +03:00
if not minetest.check_player_privs(name, { privs = true }) then
return false
end
-- code
end)
```
2018-04-03 15:11:27 +03:00
## Set ItemStacks After Changing Them
2018-04-12 03:06:07 +03:00
Notice how it's simply called an `ItemStack` in the API, not an `ItemStackRef`,
similar to `InvRef`. This is because an `ItemStack` isn't a reference - it's a
copy. Stacks work on a copy of the data rather than the stack in the inventory.
This means that modifying a stack won't actually modify that stack in the inventory.
2018-04-03 15:11:27 +03:00
Don't do this:
```lua
2018-04-03 15:11:27 +03:00
local inv = player:get_inventory()
local stack = inv:get_stack("main", 1)
stack:get_meta():set_string("description", "Partially eaten")
-- BAD! Modification will be lost
```
2018-04-03 15:11:27 +03:00
Do this:
```lua
2018-04-03 15:11:27 +03:00
local inv = player:get_inventory()
local stack = inv:get_stack("main", 1)
stack:get_meta():set_string("description", "Partially eaten")
inv:set_stack("main", 1, stack)
-- Correct! Item stack is set
```
2018-04-03 15:11:27 +03:00
The behaviour of callbacks is slightly more complicated. Modifying an itemstack you
are given will change it for the caller too, and any subsequent callbacks. However,
2018-04-03 15:11:27 +03:00
it will only be saved in the engine if the callback caller sets it.
Avoid this:
```lua
2018-09-24 19:16:00 +03:00
minetest.register_on_item_eat(function(hp_change, replace_with_item,
itemstack, user, pointed_thing)
2018-04-03 15:11:27 +03:00
itemstack:get_meta():set_string("description", "Partially eaten")
2018-09-24 19:16:00 +03:00
-- Almost correct! Data will be lost if another
-- callback cancels the behaviour
2018-04-03 15:11:27 +03:00
end)
```
2018-04-03 15:11:27 +03:00
If no callbacks cancel, then the stack will be set and the description will be updated.
If a callback cancels, then the update may be lost. It's better to do this instead:
2018-04-03 15:11:27 +03:00
```lua
2018-09-24 19:16:00 +03:00
minetest.register_on_item_eat(function(hp_change, replace_with_item,
itemstack, user, pointed_thing)
2018-04-03 15:11:27 +03:00
itemstack:get_meta():set_string("description", "Partially eaten")
2018-09-24 19:16:00 +03:00
user:get_inventory():set_stack("main", user:get_wield_index(),
itemstack)
2018-04-03 15:11:27 +03:00
-- Correct, description will always be set!
end)
```
2018-04-03 15:11:27 +03:00
If the callbacks cancel or the callback runner doesn't set the stack,
then our update will still be set.
If the callbacks or the callback runner does set the stack, then our
2018-05-01 02:00:27 +03:00
set_stack doesn't matter.