-
Notifications
You must be signed in to change notification settings - Fork 219
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add guide to using Rednet #1647
Open
MCJack123
wants to merge
3
commits into
cc-tweaked:mc-1.20.x
Choose a base branch
from
MCJack123:patch-17
base: mc-1.20.x
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from all commits
Commits
Show all changes
3 commits
Select commit
Hold shift + click to select a range
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,388 @@ | ||
--- | ||
module: [kind=guide] rednet_guide | ||
see: rednet Send and receive messages over modems. | ||
--- | ||
|
||
<!-- | ||
SPDX-FileCopyrightText: 2023 The CC: Tweaked Developers | ||
|
||
SPDX-License-Identifier: MPL-2.0 | ||
--> | ||
|
||
# Transferring information wirelessly using Rednet | ||
Being able to send data between computers is an important feature for various | ||
programs, including turtle controllers and distributed banking systems. | ||
[modem](https://tweaked.cc/peripheral/modem.html) peripheral enables computers | ||
and turtles to transmit and receive messages from other computers, and | ||
[Rednet](https://tweaked.cc/module/rednet.html) API allows sending messages | ||
directly to computers by ID, as well as adding hostname lookup and message | ||
repeating. This guide will show how to use the Rednet API to send and receive | ||
data, as well as some tips on usage. | ||
|
||
## Getting started | ||
To begin using Rednet, you will need at least two computers (or turtles) with | ||
modems attached. There are three different types of modems available: | ||
|
||
- Wireless Modems are the simplest to craft. They have a limited range - | ||
computers that are too far from each other won't be able to receive each others' | ||
messages. They also don't work across dimensions. | ||
- Ender Modems have infinite range and work across dimensions, but they require | ||
an Eye of Ender to craft. But because they're infinite, messages sent will be | ||
received by every computer on the server, which could be concerning for | ||
multiplayer servers where security is required. | ||
- Wired Modems transmit messages through a wire. They don't have limited range, | ||
but they'll only send messages to computers connected to the same wire. They can | ||
also connect remote peripherals, including inventories like chests. There are | ||
two variations: the "half width" modems connect to computers directly, while | ||
"full block" modems can connect to turtles, and can function as wires as well. | ||
|
||
To attach a modem to a computer block, simply hold shift and right-click on the | ||
computer. To attach a modem to a turtle or pocket computer, craft the computer | ||
with a wireless modem, or use the `equip` program with the modem's slot | ||
selected. If you use a wired modem, you'll also need to right-click the modem to | ||
turn it red - this will connect the computer to the network. | ||
|
||
Before using Rednet, your program needs to open it. Opening Rednet will tell the | ||
modem to start listening for messages from other computers. The | ||
[`rednet.open`](https://tweaked.cc/module/rednet.html#v:open) function takes the | ||
side of the modem to open as a string. On pocket computers, this will always be | ||
`back`. For example, if your modem is on the left side of the computer, you | ||
would use this code: | ||
|
||
```lua | ||
rednet.open("left") | ||
``` | ||
|
||
You can also use this code to open every attached modem at once: | ||
|
||
```lua | ||
peripheral.find("modem", rednet.open) -- (Ab)uses peripheral.find's filter parameter to run rednet.open for each modem found. | ||
``` | ||
|
||
Always run this function before you do anything with Rednet - otherwise, it may | ||
not function as expected. | ||
|
||
## Sending messages | ||
To send a message to another computer, use the | ||
[`rednet.send`](https://tweaked.cc/module/rednet.html#v:send) function. This | ||
function takes the ID of the computer to send to (which you can get with the | ||
`id` command), the message you want to send, and the protocol to use if required | ||
(more on protocols later). | ||
|
||
This line will send the string `"Hello, World!"` to computer ID 3. | ||
|
||
```lua | ||
rednet.send(3, "Hello, World!") | ||
``` | ||
|
||
You can send almost any type of value, including tables. You can also send | ||
variables. Here is an example which sends a table of values from a variable: | ||
|
||
```lua | ||
local message = { | ||
name = "My Message", | ||
length = 5, | ||
contents = {1, 2, 5, 4, 3} | ||
} | ||
rednet.send(3, message) | ||
``` | ||
|
||
You can even send the contents of files using | ||
[`fs.open`](https://tweaked.cc/module/fs.html#v:open): | ||
|
||
```lua | ||
local file = assert(fs.open("myfile.txt", "r")) -- Opens the file for reading, with assert to error if it fails. | ||
local data = file.readAll() -- Read the full file into a variable. | ||
file.close() -- Always close files when you're done! | ||
rednet.send(3, data) | ||
``` | ||
|
||
Note that you cannot send the file handle itself - functions can't be sent over | ||
Rednet, so the file reading functions will get removed from the table, resulting | ||
in sending an empty table. | ||
|
||
If a message needs to be sent to every computer available, | ||
[`rednet.broadcast`](https://tweaked.cc/module/rednet.html#v:broadcast) can be | ||
used. It takes the message to send, as well as the protocol if desired. | ||
|
||
```lua | ||
rednet.broadcast("Emergency message!") | ||
``` | ||
|
||
## Receiving messages | ||
Sending messages isn't really useful unless there's a way to receive them. The | ||
[`rednet.receive`](https://tweaked.cc/module/rednet.html#v:receive) function | ||
**waits** for a message to be received, and can wait for a certain amount of | ||
time, or filter for a specific protocol (again, more about that later). It | ||
returns the ID of the computer that sent the message, and then the message (and | ||
finally the protocol if specified), as *multiple return values* (NOT a table). | ||
|
||
Here is a simple example that waits for any message without a timeout: | ||
|
||
```lua | ||
local id, message = rednet.receive() | ||
``` | ||
|
||
This example will wait for 5 seconds to receive a message - if nothing is sent | ||
in 5 seconds, it will return `nil`. | ||
|
||
```lua | ||
local id, message = rednet.receive(5) | ||
if id == nil then -- Check if it timed out. | ||
error("No message received!") | ||
end | ||
``` | ||
|
||
It's a good idea to check that the message came from the right computer before | ||
processing it further. This chunk will check that the message was sent by | ||
computer 1, and if so, will print the message sent to the screen: | ||
|
||
```lua | ||
local id, message = rednet.receive() | ||
if id == 1 then -- Check for the right sender. | ||
print("Message name:", message.name) | ||
end -- Ignore it if another computer sent it. | ||
``` | ||
|
||
Usually, a server-like program will combine this with an infinite loop to | ||
constantly process requests. Here's a full example of a server with multiple | ||
commands that can be triggered: | ||
|
||
```lua | ||
while true do -- Repeat forever. | ||
local id, message = rednet.receive() | ||
if id == 1 then -- Check for the right sender. | ||
-- Check the message command, and execute it. | ||
if message == "forward" then | ||
turtle.forward() | ||
elseif message == "back" then | ||
turtle.back() | ||
elseif message == "left" then | ||
turtle.turnLeft() | ||
elseif message == "right" then | ||
turtle.turnRight() | ||
elseif message == "dig" then | ||
turtle.dig() | ||
elseif message == "quit" then | ||
break -- This will break out of the infinite loop, quitting the program. | ||
end -- Do nothing on an invalid command. | ||
end -- Ignore it if another computer sent it. | ||
end | ||
``` | ||
|
||
## Filtering messages with protocols | ||
Checking for messages by computer ID is a quick way to check that a message is | ||
intended for this server, but sometimes multiple computers need to send messages | ||
to one computer. Protocols allow you to add a special "tag" to a message, which | ||
is used to filter messages. This will prevent computers from accidentally | ||
receiving messages that weren't meant for them. | ||
|
||
To add a protocol to a message, simply pass it as the third argument to | ||
`rednet.send`: | ||
|
||
```lua | ||
rednet.send(3, message, "myProtocol") | ||
``` | ||
|
||
To keep your protocol unique to yourself, while also being identifiable, it's | ||
recommended to use a protocol with your name (or a reverse URL) and a program | ||
name separated by a dot: | ||
|
||
```lua | ||
rednet.send(3, message, "SquidDev.better-chat") | ||
-- if you own a relevant URL, use it instead: | ||
rednet.send(3, message, "cc.squiddev.better-chat") | ||
``` | ||
|
||
On the receiving end, messages can be filtered as the last argument to | ||
`rednet.receive`: | ||
|
||
```lua | ||
local id, message = rednet.receive("cc.squiddev.better-chat") | ||
-- if you want a timeout, put it first: | ||
local id, message = rednet.receive(5, "cc.squiddev.better-chat") | ||
``` | ||
|
||
The protocol is returned as an additional third return value, which can be used | ||
to do manual filtering if more than one protocol is necessary: | ||
|
||
```lua | ||
local id, message, protocol = rednet.receive() | ||
if protocol == "cc.squiddev.better-chat.message" then | ||
-- ... | ||
elseif protocol == "cc.squiddev.better-chat.query" then | ||
-- ... | ||
end | ||
``` | ||
|
||
## Dynamic host ID lookup | ||
Sometimes, you may not know the exact computer ID to send to - you only know | ||
that you need to send a certain protocol. Rednet features a dynamic host lookup | ||
procedure, which lets servers advertise that they listen to a certain protocol. | ||
Other computers can broadcast a message asking what servers support a protocol, | ||
and any computers which "host" it will respond with their ID. These IDs can be | ||
used directly in `rednet.send` calls. This makes setting up large networks | ||
simpler, as you don't need to hard-code the computer ID into each and every | ||
computer. It also allows creating a list of computers that support a protocol - | ||
for example, creating a server list for a chat program. | ||
|
||
To advertise a protocol, use the | ||
[`rednet.host`](https://tweaked.cc/module/rednet.html#v:host) function. This | ||
function takes the protocol to host, as well as a name to call the computer on | ||
the network. | ||
|
||
```lua | ||
rednet.host("cc.squiddev.better-chat", "super-cool-computer") | ||
``` | ||
|
||
Once your program finishes, remember to use the | ||
[`rednet.unhost`](https://tweaked.cc/module/rednet.html#v:unhost) function to | ||
stop advertising the protocol: | ||
|
||
```lua | ||
rednet.unhost("cc.squiddev.better-chat") | ||
``` | ||
|
||
The [`rednet.lookup`](https://tweaked.cc/module/rednet.html#v:lookup) function | ||
takes a protocol to search for, and optionally a name to find, and it returns | ||
*multiple return values* with each computer ID found, or `nil` if there were no | ||
matches. Here's a basic example looking for any computer hosting a protocol: | ||
|
||
```lua | ||
local ids = {rednet.lookup("cc.squiddev.better-chat")} -- Wraps the multiple return values into a single table. | ||
``` | ||
|
||
This will look for a computer with a specific name, erroring if it's not found: | ||
|
||
```lua | ||
local id = rednet.lookup("cc.squiddev.better-chat", "super-cool-computer") -- Does not create a table - takes the first return value directly. | ||
if id == nil then error("No route to host") end | ||
``` | ||
|
||
This example will scan for a protocol, and pings each computer which hosts, but | ||
errors if there are no computers found. | ||
|
||
```lua | ||
local ids = {rednet.lookup("cc.squiddev.better-chat")} | ||
if #ids == 0 then -- Checks if the table is empty. | ||
error("No computers found!") | ||
else | ||
for _, id in ipairs(ids) do -- Loop over all IDs in the table. | ||
print("Found computer", id) -- Print the ID to the screen. | ||
rednet.send(id, "PING!", "cc.squiddev.better-chat") -- Send a message to the computer. | ||
end | ||
end | ||
``` | ||
|
||
## Security over Rednet | ||
One important thing to be aware of, especially on multiplayer servers, is that | ||
Rednet is not a secure protocol. Anyone can intercept messages sent over it | ||
(unless the message is sent through a wired modem), with the contents available | ||
in plain text. Furthermore, the ID of a message can be spoofed, and a protocol | ||
doesn't guarantee that the message is in the right format. | ||
|
||
If security and authentication are necessary for your application, you should | ||
add encryption to messages. This requires extra code from other people, and is | ||
often more complex than simple send/receive message calls. But for things like | ||
banking and secure chat, encryption is necessary to keep communications safe. | ||
|
||
The simplest way to use encryption is through a library like [ECNet2], which | ||
uses protocols to create two-way encrypted pipes (or "sockets") between a client | ||
and server. See the examples there for more info on how to use it. | ||
|
||
You can also use encryption primitives directly. There are various algorithms | ||
that each contribute to the security of a message. They have different purposes, | ||
so you'll need to pick and choose which ones apply to your use case. | ||
|
||
> [Expert zone][!WARNING] | ||
> | ||
> These terms may be hard to understand for beginners. Cryptography is a very | ||
> complex field, and uses a lot of terms that may be unfamiliar to novices. This | ||
> guide attempts to boil it down to more understandable terms, but even so, it | ||
> may not be enough for someone new to programming to understand. If in doubt, | ||
> use ECNet, or cross your fingers that nobody will peek at your messages. | ||
|
||
- SHA-256 is a hashing algorithm, made popular by its extensive use in Bitcoin. | ||
SHA-256 takes a large string of data, and mashes it up in a semi-random way to | ||
create a 32-byte string/number that represents the data, called a "hash" (or | ||
"digest"). The hash cannot easily be reversed into its original string, but the | ||
same string will create the same hash every time. SHA-256 is useful for | ||
applications where you need to know whether two strings are the same without | ||
storing the actual string, such as checking passwords. SHA-256 is succeeded by | ||
SHA-512, the SHA-3 family of algorithms, and the BLAKE3 family of algorithms, | ||
which are all more secure and have a larger hash than SHA-256; but SHA-256 is | ||
much simpler to implement and faster in CC, as well as still being unbroken, so | ||
it remains the most popular choice for hashing algorithms. | ||
- HMAC is a message authentication algorithm, used for checking the validity and | ||
sender of a message, and is based on a hashing algorithm. HMAC takes a string | ||
message, and a secret key, and creates an "authentication tag" from those two. | ||
The authentication tag is then sent next to the message, but the key is kept | ||
safe and out of the message. On the other end, the receiver can verify the | ||
message by creating its own authentication tag from the message and its copy of | ||
the key, and compares that with the authentication tag with the message. This | ||
ensures that the message wasn't tampered with, and that the message was sent by | ||
the original sender, because changing either the message or key will create a | ||
different tag on the other side. However, it does not hide the contents of the | ||
message. Poly1305 is a more modern and faster alternative that's often used | ||
instead of HMAC, and uses a special algorithm instead of hashing. | ||
- PBKDF2 is a key generation algorithm, which makes a random encryption key from | ||
a password. PBKDF2 takes a password and a random string called a "salt", and | ||
generates a key that can be used for encryption. It uses a large number of | ||
cycles of an algorithm like HMAC to create a key that's random enough to not be | ||
crackable, which also increases the amount of time to try each password. PBKDF2 | ||
is useful not only for encryption using a password, but also for checking | ||
passwords, as it makes it even more difficult to reverse the password hash. | ||
- ChaCha20 is a symmetric encryption algorithm, which uses a shared key to hide | ||
the contents of a message. It takes the message, a key, and 12 bytes of random | ||
"salt" data, and creates a scrambled string that can only be decoded with the | ||
same key and salt. The salt should be stored next to the message, but the key | ||
should be kept safe and never sent anywhere. AES is a popular alternative to | ||
ChaCha20, but it's more complex and slower, which makes it a poor fit for | ||
ComputerCraft. ChaCha20 is often paired with Poly1305 to make a type of | ||
encryption known as *Authenticated Encryption with Associated Data* (AEAD), | ||
which both hides a message *and* authenticates it with the original sender, and | ||
it can even authenticate plain text in the same operation. | ||
- Curve25519/X25519 is an elliptic curve (ECC) asymmetric cryptographic | ||
algorithm, which uses separate publicly shareable and private keys to exchange | ||
data in a trusted way. Its most common usage is for Elliptic Curve | ||
Diffie-Hellman (ECDH), which is able to create a shared symmetric key by only | ||
sending the public keys of each computer. This key can then be used for ChaCha20 | ||
encryption. X25519 and ECDH are important for encryption because they allow | ||
computers to create a secret key, without sending enough data for other | ||
computers to see the secret as well. To use ECDH, first generate a public and | ||
private keypair on both sides; then send just the *public* key from both | ||
computers to each other. Once both computers have each others' public keys, run | ||
ECDH using *this* computer's *private* key, and the *other* computer's *public* | ||
key. Through the work of magic, both computers will have the same secret key | ||
despite never sharing it directly, and no other computer will be able to create | ||
it because they don't have a private key. | ||
- Ed25519 is a variant of X25519 that allows *signing* messages. Signing is like | ||
message authentication, but uses asymmetric public and private keys to create | ||
the tag. Signing involves using the private key on the message to create a | ||
signature, and verification uses the public key on the signature to check it | ||
with the original message. Signing is used heavily in the real world in HTTPS, | ||
which uses "certificates" signed by higher powers to encrypt the connection. The | ||
signature ensures that the website can be trusted and is who it says it is. | ||
Signing is useful anywhere where authenticating the source and validity of a | ||
message is important, but it's critical that the receiver can't create its own | ||
authentication tags. | ||
|
||
Some popular encryption primitive libraries include | ||
[Anavrins's SHA256/HMAC/PBKDF2 library], [PG231's ECC library], and [CCryptoLib] | ||
which is used in ECNet2. | ||
|
||
## Conclusion | ||
Remote tasks become a lot easier when computers can communicate together. The | ||
Rednet API makes it possible to send messages between computers simply. | ||
`rednet.send` and `rednet.receive` are the building blocks for transmitting | ||
messages over modems. Protocols help smoothen out these functions by filtering | ||
out the messages that aren't important. Host lookup makes it possible to send to | ||
computers without needing to know their IDs directly. And despite Rednet being | ||
an insecure protocol, there are many ways to fortify your connection against | ||
unwanted spying and manipulation. These tools are fundamental to any program | ||
that relies on remote communication. | ||
|
||
[ECNet2]: https://github.com/migeyel/ecnet/ | ||
[Anavrins's SHA256/HMAC/PBKDF2 library]: https://pastebin.com/6UV4qfNF | ||
[PG231's ECC library]: https://www.computercraft.info/forums2/index.php?/topic/29803-elliptic-curve-cryptography/ | ||
[CCryptoLib]: https://github.com/migeyel/ccryptolib/ |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We use the same markdown processor as the rest of the docs, so it should be possible to use
[`modem`]
here instead.