74 Commits

Author SHA1 Message Date
bc5617407d Update CHANGELOG.md 2024-12-02 10:40:51 +02:00
5ae06f4d37 Update man page 2024-12-02 10:35:00 +02:00
233031cdcb Update README.md 2024-12-02 10:31:38 +02:00
ebf5172cd0 Update BUILDING.md 2024-12-02 10:13:38 +02:00
0b72e4d17d Update README.md 2024-12-02 09:56:06 +02:00
af70f87bc1 Update README.md 2024-12-02 09:52:30 +02:00
53fd703911 Rename INSTALL.md to BUILDING.md 2024-12-02 09:42:44 +02:00
c0d28bcb17 Rename INSTALL.md to BUILDING.md 2024-12-02 09:42:11 +02:00
edf8344983 Add "-Wno-gnu-zero-variadic-macro-arguments" flag to suppress Clang warnings about GNU macro extensions 2024-12-02 09:37:08 +02:00
a502204e26 Remove unused headers 2024-12-02 09:34:48 +02:00
7162bc6fe6 - add log_error macro
- rename "flag_connection_alive" variable back to "global_connection_alive"
- return exit code from run_terminal_mode()
- rcon_command(): add error messages
2024-12-02 08:55:18 +02:00
f270a485b5 Remove url from version string 2024-12-02 07:40:34 +02:00
df66816bee Change prefix of flag variables from 'global' to 'flag' 2024-12-02 07:37:12 +02:00
5c7ab407d7 Patch of various fixes, cleanups and unused code removals:
- add MAX_COMMAND_LENGTH to define maximum command length
 - print auth failed message to stderr instead of stdout
 - remove unused net_send() function
 - remove unused net_clean_incoming() function
 - rewrite net_send_packet() function
 - net_recv_packet(): change the type of variable "ret" from int to ssize_t
 - net_recv_packet(): fail immediately if the packet size is out of spec
 - packet_print(): rename variable "def_color" to "default_color"
 - packet_print(): remove unecessary casts
 - packet_build(): use MAX_COMMAND_LENGTH
 - packet_build(): be more explicit in calculation of packet.size
 - packet_build(): use memcpy() instead of strncpy()
 - cast second argument of send()/recv() calls to (char *) so Windows is happy
 - rcon_auth(): change the return type from int to bool
 - run_terminal_mode(): use MAX_COMMAND_LENGTH
2024-12-02 07:32:07 +02:00
c83d96cc91 Add "$(EXENAME).exe" to clean rule so Windows executables are also cleaned up 2024-12-02 06:14:34 +02:00
427fd206ca Oops, uncomment struct field 2024-11-11 23:49:18 +02:00
1a4010cbba Change MAX_PACKET_SIZE and DATA_BUFFSIZE, add notes about packet structure 2024-11-11 22:22:45 +02:00
4488127350 Use fixed width integer types in rcon packet structure 2024-11-10 16:25:19 +02:00
6b563df23d Update README.md 2024-11-09 20:53:55 +02:00
5a2dcf41ac Update README.md 2024-11-09 20:51:36 +02:00
7d3f3c1d61 Update README.md 2024-11-09 20:46:53 +02:00
fc040ce5ea Change maximum packet size to correct value (4096 -> 4106) 2024-11-09 15:22:42 +02:00
489306d4a2 Add windows batch scripts 2024-11-09 13:48:23 +02:00
a8e2a9349e Exit with appropriate return code if password is not provided, fixes #87 2024-11-07 21:39:37 +02:00
deed43ad61 Makefile macOS fix, resolves #82, resolves #102
- Remove "ginstall" as "install" replacement on macOS
- Remove "-D" flag from "install" parameters
2024-11-07 17:14:25 +02:00
aa933d2c1f Add support for Valve style authentication, fixes #106 2024-11-07 14:59:31 +02:00
accae57e4b Remove ".travis.yml" 2024-11-07 14:21:31 +02:00
fa25cde79c Modify compilation flags:
- Remove "-s" flag
- Change stack protector mode from "strong" to "all"
- Change optimization level from "-Os" to "-O2"
2024-11-07 14:06:48 +02:00
e96b2eff6e Remove broken Travis CI links and update package list 2024-11-07 13:54:40 +02:00
b5951e9634 Merge pull request #83 from Tiiffi/develop
Merge develop to master
2021-10-30 22:34:42 +03:00
7b8ea2bf39 Update copyright year 2021-10-30 22:30:12 +03:00
05aaff88d4 Update version information, changelog and readme 2021-10-30 22:21:25 +03:00
fca278e092 - Quit gracefully when Ctrl-D or Ctrl-C is pressed
- Remove "exit" and "quit" as quit commands
2021-10-30 22:16:29 +03:00
b3147ebe43 Fix erroneous string length check 2021-02-15 04:31:07 +02:00
48c065c304 Use setvbuf() instead of fflush() 2021-02-15 03:29:44 +02:00
29a1c99f82 Merge pull request #53 from AddisonG/master
Fix compiler warning
2021-02-12 04:23:54 +02:00
bf11460a0d Merge pull request #39 from kabiroberai/master
Flush stdout when needed
2021-02-08 22:51:54 +02:00
3e8acd5e42 Merge pull request #62 from jbaldus/master
Fixes typo in ANSI escape sequence for LCYAN
2021-02-08 05:16:11 +02:00
ada14bb4d9 Fixes typo in ANSI escape sequence for LCYAN 2021-01-13 19:22:02 -05:00
336f528668 Fixed compiler bug
This fixes the compiler issue:

```
mcrcon.c: In function ‘packet_build’:
mcrcon.c:576:2: warning: ‘strncpy’ specified bound 4096 equals destination size [-Wstringop-truncation]
  strncpy(packet.data, s1, DATA_BUFFSIZE);
  ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
```
2020-08-12 14:10:58 +10:00
58a7a86bc3 Flush stdout when needed
This is required on certain machines (eg the iSH emulator on iOS)
2020-01-31 17:26:27 +05:30
b02201d689 Merge pull request #37 from Tiiffi/develop
Update CHANGELOG.md
2020-01-02 03:14:12 +02:00
4cff2fe537 Update CHANGELOG.md 2020-01-02 03:11:50 +02:00
0d11c6a4e9 Update README.md 2020-01-02 02:40:05 +02:00
9c68d9cdd4 Update README.md 2020-01-02 02:30:12 +02:00
6d49968df3 Update README.md 2020-01-02 02:28:52 +02:00
b118b7b7b9 Change year in copyright texts and manual page 2020-01-02 02:22:44 +02:00
253fb70a72 Merge branch 'develop' 2020-01-02 02:15:27 +02:00
f351fb134e - Deprecate -i flag for terminal mode
- Use strcasecmp() instead strcmp() to check exit command
2019-12-22 16:27:36 +02:00
af5e88b528 Update Makefile 2019-12-22 00:05:31 +02:00
83cdaae362 Update version number 2019-12-21 23:13:19 +02:00
876201fe5a Add workaround to prevent server-side bug.
https://bugs.mojang.com/browse/MC-154617
2019-12-21 23:12:39 +02:00
3c071c0c45 - Add exit string detection
- Stylistic change in getopt loop
2019-12-21 22:25:08 +02:00
d2ebbd8f12 Update various document files 2019-12-21 22:16:34 +02:00
cffa73d223 Remove unused functions 2019-12-20 21:02:21 +02:00
c62d5c8a70 Update usage and version texts 2019-12-20 20:58:40 +02:00
8b75ddf4bf Indentation style change 2019-12-20 19:56:05 +02:00
4f3a455095 Add more validation to mcrcon_parse_seconds() 2019-12-19 17:34:19 +02:00
5fb20c2b83 Update mcrcon.1 and README.md 2019-12-19 17:32:47 +02:00
0c48fbb847 Remove unused option characters 2019-12-18 19:34:08 +02:00
fcc9d6d09b Tabs are evil 2019-12-18 19:19:30 +02:00
018592abc5 Code cleanups 2019-12-18 19:16:06 +02:00
de926b8c26 Merge branch 'develop' of https://github.com/Tiiffi/mcrcon into develop 2019-12-18 19:08:21 +02:00
0f796b6bbc Tidy code formatting 2019-12-18 19:06:37 +02:00
f9d9a02348 Update README.md 2019-12-18 18:29:31 +02:00
a8863e1fcc Merge branch 'develop' of https://github.com/Tiiffi/mcrcon into develop 2019-12-18 18:19:50 +02:00
83457075b9 Make command throttling argument parsing more robust 2019-12-18 18:19:08 +02:00
2f1f925ac1 Make command throttling more robust 2019-12-18 18:02:20 +02:00
8ff710e231 Update README.md 2019-12-18 18:00:45 +02:00
7737bc2a32 Update mcrcon.1 2019-12-18 18:00:31 +02:00
52e717a92b Update CHANGELOG.md 2019-12-18 18:00:11 +02:00
8fdda295e2 Update mcrcon.1 2019-11-19 09:47:05 +02:00
0b1853dccc Update INSTALL.md 2019-11-19 09:17:20 +02:00
d338537e23 Update README.md 2019-11-19 09:16:11 +02:00
11 changed files with 481 additions and 433 deletions

View File

@ -1,20 +0,0 @@
language: c
sudo: false
dist: trusty
cache:
ccache: true
compiler:
- gcc
- clang
env:
global:
- EXTRAFLAGS='-v'
- PREFIX="${HOME}"
install:
- 'true'
script:
- make
- ./mcrcon -h
- ./mcrcon -v
- make install
- make uninstall

29
BUILDING.md Normal file
View File

@ -0,0 +1,29 @@
Building and installing
-----------------------
#### Requirements & dependencies:
- GCC compatible compiler
- POSIX getopt support from **<unistd.h>**
---
#### Compiling
cc -std=gnu99 -Wpedantic -Wall -Wextra -Wno-gnu-zero-variadic-macro-arguments -O2 -o mcrcon mcrcon.c
>[!NOTE]
>If you are compiling on Windows remember to link with winsock by adding `-lws2_32` to your compiler command line.
---
Or you can run **make**
make - compiles mcrcon
make install - installs compiled binaries and manpage to the system
make uninstall - removes binaries and manpage from the system
file install locations:
/usr/local/bin/mcrcon
/usr/local/share/man/man1/mcrcon.1
Makefile **install** and **uninstall** rules are not available on Windows.

View File

@ -1,5 +1,31 @@
#### Version history: #### Version history:
###### 0.7.3
- Add support to Valve style rcon authentication
- Change maximum packet size to correct value (4096 -> 4106)
- Print auth failed message to stderr instead of stdout
- Fail immediately if received packet size is out of spec
- Return proper exit code from run_terminal_mode()
- Add error messages to rcon_command() function
###### 0.7.2
- Quit gracefully when Ctrl-D or Ctrl+C is pressed
- Remove "exit" and "quit" as quitting commands
* these are actual rcon commands on some servers
- Suppress compiler warning (strncpy)
- Fix erroneous string length in packet building function
- Fix typo in ANSI escape sequence for LCYAN
- Make stdout and stderr unbuffered
###### 0.7.1
- Deprecate `-i` flag for invoking terminal mode
- Add workaround to prevent server-side bug.
* https://bugs.mojang.com/browse/MC-154617
###### 0.7.0
- Add -w option for rcon command throttling
* Thanks HorlogeSkynet @ Github
###### 0.6.2 ###### 0.6.2
- Set default address to localhost - Set default address to localhost

View File

@ -1,22 +0,0 @@
Compiling and installing
------------------------
Only dependency is C library and POSIX getopt support.
Compiling with GCC or CLANG:
cc -std=gnu99 -Wpedantic -Wall -Wextra -Os -s -o mcrcon mcrcon.c
Note: on Window remember to link with winsockets by adding "-lws2_32" to your compiler command line.
Or you can just run "make":
make - compiles mcrcon
make install - installs compiled binaries and manpage to the system
make uninstall - removes binaries and manpage from the system
file install locations:
/usr/local/bin/mcrcon
/usr/local/share/man/man1/mcrcon.1
Makefile "install" and "uninstall" rules are disabled on windows.

View File

@ -1,4 +1,4 @@
Copyright (c) 2012-2019, Tiiffi <tiiffi at gmail> Copyright (c) 2012-2024, Tiiffi <tiiffi at gmail>
This software is provided 'as-is', without any express or implied This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages warranty. In no event will the authors be held liable for any damages
@ -18,4 +18,3 @@ freely, subject to the following restrictions:
3. This notice may not be removed or altered from any source 3. This notice may not be removed or altered from any source
distribution. distribution.

View File

@ -1,7 +1,10 @@
# if you want to cross compile # if you want to cross compile:
# export PATH=$PATH:/path/to/compiler/bin # export PATH=$PATH:/path/to/compiler/bin
# export CROSS_COMPILE=arm-none-linux-gnueabi- # export CROSS_COMPILE=arm-none-linux-gnueabi-
# make # make
#
# Windows cross compile:
# x86_64-w64-mingw32-gcc -std=gnu99 -Wall -Wextra -Wpedantic -O2 -fstack-protector-all -o mcrcon.exe mcrcon.c -lws2_32
EXENAME = mcrcon EXENAME = mcrcon
PREFIX ?= /usr/local PREFIX ?= /usr/local
@ -10,9 +13,9 @@ INSTALL = install
LINKER = LINKER =
RM = rm -v -f RM = rm -v -f
CC = cc CC ?= gcc
CFLAGS = -std=gnu99 -Wall -Wextra -Wpedantic -Os -s CFLAGS = -std=gnu99 -Wall -Wextra -Wpedantic -Wno-gnu-zero-variadic-macro-arguments -O2
EXTRAFLAGS ?= -fstack-protector-strong EXTRAFLAGS ?= -fstack-protector-all
ifeq ($(OS), Windows_NT) ifeq ($(OS), Windows_NT)
LINKER = -lws2_32 LINKER = -lws2_32
@ -20,11 +23,6 @@ ifeq ($(OS), Windows_NT)
RM = cmd /C del /F RM = cmd /C del /F
endif endif
ifeq ($(shell uname), Darwin)
INSTALL = ginstall
CFLAGS = -std=gnu99 -Wall -Wextra -Wpedantic -Os
endif
.PHONY: all .PHONY: all
all: $(EXENAME) all: $(EXENAME)
@ -34,8 +32,8 @@ $(EXENAME): mcrcon.c
ifneq ($(OS), Windows_NT) ifneq ($(OS), Windows_NT)
.PHONY: install .PHONY: install
install: install:
$(INSTALL) -vD $(EXENAME) $(DESTDIR)$(PREFIX)/bin/$(EXENAME) $(INSTALL) -v $(EXENAME) $(DESTDIR)$(PREFIX)/bin/$(EXENAME)
$(INSTALL) -vD -m 0644 mcrcon.1 $(DESTDIR)$(PREFIX)/share/man/man1/mcrcon.1 $(INSTALL) -v -m 0644 mcrcon.1 $(DESTDIR)$(PREFIX)/share/man/man1/mcrcon.1
@echo "\nmcrcon installed. Run 'make uninstall' if you want to uninstall.\n" @echo "\nmcrcon installed. Run 'make uninstall' if you want to uninstall.\n"
.PHONY: uninstall .PHONY: uninstall
@ -46,4 +44,4 @@ endif
.PHONY: clean .PHONY: clean
clean: clean:
$(RM) $(EXENAME) $(RM) $(EXENAME) $(EXENAME).exe

View File

@ -1,33 +1,42 @@
# mcrcon # mcrcon
mcrcon is console based Minecraft [rcon](https://developer.valvesoftware.com/wiki/Source_RCON_Protocol) client for remote administration and server maintenance scripts. mcrcon is a command-line [rcon](https://developer.valvesoftware.com/wiki/Source_RCON_Protocol) client intended for remote server administration and maintenance automation.
Though originally developed for Minecraft servers, it also works with a variety of other servers using the Valve or Minecraft-style rcon protocol.
--- ---
### Installing: ### Installing
##### via packet manager: ##### Binary releases
See https://pkgs.org/download/mcrcon for available packages in various Linux distros.
- Gentoo Linux: https://packages.gentoo.org/packages/games-util/mcrcon Pre-built binaries are provided for Linux and Windows: https://github.com/Tiiffi/mcrcon/releases/latest
- Arch Linux: https://aur.archlinux.org/packages/mcrcon/
##### from sources: ##### Via package manager
See https://pkgs.org/download/mcrcon and https://repology.org/project/mcrcon/packages for available packages in various Linux distros (note that some packages might be outdated).
- Fedora: https://packages.fedoraproject.org/pkgs/mcrcon/mcrcon/
- Gentoo: https://packages.gentoo.org/packages/games-util/mcrcon
- Arch: https://aur.archlinux.org/packages/mcrcon/
- NixOS: https://search.nixos.org/packages?show=mcrcon
- Snapcraft: https://snapcraft.io/mcrcon-nsg
- Scoop: https://scoop.sh/#/apps?q=mcrcon
##### Building from sources
```sh ```sh
git clone https://github.com/Tiiffi/mcrcon.git git clone https://github.com/Tiiffi/mcrcon.git
cd mcrcon cd mcrcon
make make
# install is optional
sudo make install sudo make install
``` ```
Check [INSTALL.md](INSTALL.md) for more details. _Check [BUILDING.md](BUILDING.md) for more details._
You can also download precompiled binaries*: https://github.com/Tiiffi/mcrcon/releases/latest
<sub>*At the moment binaries are provided for Linux and Windows.</sub>
--- ---
### Usage: ### Usage
mcrcon [OPTIONS] [COMMANDS] mcrcon [OPTIONS] [COMMANDS]
Sends rcon commands to Minecraft server. Sends rcon commands to Minecraft server.
@ -41,9 +50,9 @@ Option:
-s Silent mode -s Silent mode
-c Disable colors -c Disable colors
-r Output raw packets -r Output raw packets
-w Wait for specified duration (seconds) between each command (1 - 600s)
-h Print usage -h Print usage
-v Version information -v Version information
-w Wait for specified duration (seconds) between each command
``` ```
Server address, port and password can be set using following environment variables: Server address, port and password can be set using following environment variables:
``` ```
@ -56,35 +65,44 @@ MCRCON_PASS
- Command-line options will override environment variables - Command-line options will override environment variables
- Rcon commands with spaces must be enclosed in quotes - Rcon commands with spaces must be enclosed in quotes
Example: ###### Example:
```mcrcon -H my.minecraft.server -p password "say Server is restarting!" save-all stop``` Send three commands ("say", "save-all", "stop") and wait five seconds between the commands:
```sh
mcrcon -H my.minecraft.server -p password -w 5 "say Server is restarting!" save-all stop
```
---
> [!TIP]
>Enable RCON on Minecraft server by adding following lines to [```server.properties```](https://minecraft.gamepedia.com/Server.properties) configuration file.
>```
>enable-rcon=true
>rcon.port=25575
>rcon.password=your_rcon_pasword
>```
--- ---
##### Enable rcon on server ### Contact
Remember to enable rcon by adding following lines to [```server.properties```](https://minecraft.gamepedia.com/Server.properties) file.
```
enable-rcon=true
rcon.port=25575
rcon.password=your_rcon_pasword
```
---
##### Contact:
* WWW: https://github.com/Tiiffi/mcrcon/ * WWW: https://github.com/Tiiffi/mcrcon/
* MAIL: tiiffi at gmail * MAIL: tiiffi+mcrcon at gmail
* IRC: tiiffi @ quakenet * ISSUES: https://github.com/Tiiffi/mcrcon/issues/
* BUG REPORTS: https://github.com/Tiiffi/mcrcon/issues/
> [!TIP]
>When reporting issues, please provide the following information:
>
>- Version of mcrcon: Please specify the precise version number
>- Game: Indicate the specific game server you're using (e.g., Minecraft, Valve Source Engine game, ARK, ...)
>- Server version: Provide the exact version of the game server
>- Mods and Extensions: List all mods and extensions used, including their versions
>- Issue Description: Clearly describe the problem you're encountering and the expected behavior.
>- Steps to reproduce
>
>If you're tech-savvy, consider providing a packet capture file (PCAP). Remember to use a fake password.
--- ---
### License ### License
This project is licensed under the zlib License - see the [LICENSE](LICENSE) file for details. This project is licensed under the zlib License - see the [LICENSE](LICENSE) file for details.
---
<sub>Master:</sub> ![Master build](https://api.travis-ci.org/Tiiffi/mcrcon.svg?branch=master)
<sub>Develop:</sub> ![Develop build](https://api.travis-ci.org/Tiiffi/mcrcon.svg?branch=develop)

31
create_shortcut.bat Normal file
View File

@ -0,0 +1,31 @@
@echo off
@cls
@set /p host="Enter host (default: "127.0.0.1"): "
@if "%host%"=="" set host=127.0.0.1
@set /p port="Enter port (default: 25575): "
@if "%port%"=="" set port=25575
@set /p passwd="Enter password: "
@if "%passwd%"=="" set passwd=
set name=connect_%host%-%port%
@set /p name="Enter shortcut name (default: "%name%.bat"): "
@if "%name%"=="" set name=connect_%host%-%port%
set command=@mcrcon.exe -t -H %host% -P %port% -p %passwd%
@echo %command% >> %name%.bat
@echo.
@echo Command: "%command%"
@echo.
@echo Shortcut "%name%.bat" created!
@echo.
@set "host="
@set "port="
@set "passwd="
@pause

29
launch.bat Normal file
View File

@ -0,0 +1,29 @@
@echo off
@cls
@if not exist mcrcon.exe (
@echo ERROR: Cannot find "mcrcon.exe". Bailing out!
@echo.
@pause
@exit
)
@set /p host="Enter host (default: 127.0.0.1): "
@if "%host%"=="" set host=127.0.0.1
@set /p port="Enter port (default: 25575): "
@if "%port%"=="" set port=25575
@set /p passwd="Enter password: "
@if "%passwd%"=="" set passwd=
@echo.
mcrcon.exe -t -H %host% -P %port% -p %passwd%
@echo.
@set "host="
@set "port="
@set "passwd="
@pause

View File

@ -1,7 +1,7 @@
.\" Process this file with .\" Process this file with
.\" groff -man -Tascii mcrcon.1 .\" groff -man -Tascii mcrcon.1
.\" .\"
.TH MCRCON 1 "October 2019" "Version 0.6.2" .TH MCRCON 1 "November 2024" "Version 0.7.3"
.SH NAME .SH NAME
mcrcon \- send rcon commands to a Minecraft server mcrcon \- send rcon commands to a Minecraft server
.SH SYNOPSIS .SH SYNOPSIS
@ -11,7 +11,7 @@ options
commands commands
.B ] .B ]
.SH DESCRIPTION .SH DESCRIPTION
mcrcon is Minecraft rcon client for remote administration and server maintenance scripts. mcrcon is a command-line rcon client intended for remote server administration and maintenance automation.
.SH OPTIONS .SH OPTIONS
.IP -H .IP -H
Server address (default: localhost) Server address (default: localhost)
@ -27,17 +27,17 @@ Silent mode
Disable colors Disable colors
.IP -r .IP -r
Output raw packets Output raw packets
.IP -w
Wait for specified duration (seconds) between each command (1 - 600s)
.IP -h .IP -h
Print usage Print usage
.IP -v .IP -v
Output version information Output version information
.IP -w
Wait for specified duration (seconds) between each command
.PP .PP
Commands with spaces must be enclosed in quotes. Commands with spaces must be enclosed in quotes.
.br .br
mcrcon will start in terminal mode if no commands are given. mcrcon will start in terminal mode if no commands are given.
.SH ENVIRONMENTAL VARIABLES .SH ENVIRONMENT VARIABLES
Server address, port and password can be set with following environment variables: Server address, port and password can be set with following environment variables:
.PP .PP
\fBMCRCON_HOST \fBMCRCON_HOST
@ -58,9 +58,9 @@ Send "weather clear" command to server using custom port 1337
\fBmcrcon\fR -H my.minecraft.server -P 1337 -p password "weather clear" \fBmcrcon\fR -H my.minecraft.server -P 1337 -p password "weather clear"
.RE .RE
.PP .PP
Send three commands to server ("say", "save-all" and "stop"), and wait 2 seconds between them Send three commands ("say", "save-all", "stop") and wait five seconds between the commands.
.RS .RS
\fBmcrcon\fR -H my.minecraft.server -p password -w 2 "say Server is restarting!" save-all stop \fBmcrcon\fR -H my.minecraft.server -p password -w 5 "say Server is restarting!" save-all stop
.RE .RE
.SH BUGS .SH BUGS
Bugs can be reported to \fBtiiffi+mcrcon at gmail\fR or \fBhttps://github.com/Tiiffi/mcrcon/issues/\fR Report bugs to \fBtiiffi+mcrcon at gmail\fR or \fBhttps://github.com/Tiiffi/mcrcon/issues/\fR

622
mcrcon.c
View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2012-2019, Tiiffi <tiiffi at gmail> * Copyright (c) 2012-2024, Tiiffi <tiiffi at gmail>
* *
* This software is provided 'as-is', without any express or implied * This software is provided 'as-is', without any express or implied
* warranty. In no event will the authors be held liable for any damages * warranty. In no event will the authors be held liable for any damages
@ -26,27 +26,22 @@
#include <stdint.h> #include <stdint.h>
#include <stdbool.h> #include <stdbool.h>
#include <string.h> #include <string.h>
#include <strings.h>
#include <signal.h> #include <signal.h>
#include <errno.h> #include <errno.h>
#include <unistd.h> #include <unistd.h>
#ifdef _WIN32
// for name resolving on windows
// enable this if you get compiler whine about getaddrinfo() on windows
//#define _WIN32_WINNT 0x0501
#include <ws2tcpip.h> #ifdef _WIN32
#include <winsock2.h> #include <winsock2.h>
#include <windows.h> #include <windows.h>
#include <ws2tcpip.h>
#else #else
#include <sys/types.h>
#include <sys/socket.h> #include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <netdb.h> #include <netdb.h>
#endif #endif
#define VERSION "0.6.2" #define VERSION "0.7.3"
#define IN_NAME "mcrcon" #define IN_NAME "mcrcon"
#define VER_STR IN_NAME" "VERSION" (built: "__DATE__" "__TIME__")" #define VER_STR IN_NAME" "VERSION" (built: "__DATE__" "__TIME__")"
@ -56,61 +51,59 @@
#define RCON_AUTH_RESPONSE 2 #define RCON_AUTH_RESPONSE 2
#define RCON_PID 0xBADC0DE #define RCON_PID 0xBADC0DE
// a bit too big perhaps? #define MAX_COMMAND_LENGTH 4096
#define DATA_BUFFSIZE 4096 #define DATA_BUFFSIZE MAX_COMMAND_LENGTH + 2 // plus two null terminators
#define MAX_PACKET_SIZE 4106 // id (4) + cmd (4) + DATA_BUFFSIZE
#define MIN_PACKET_SIZE 10 // id (4) + cmd (4) + two empty strings (2)
#define log_error(fmt, ...) fprintf(stderr, fmt, ##__VA_ARGS__);
// rcon packet structure // rcon packet structure
typedef struct _rc_packet { typedef struct {
int size; int32_t size;
int id; int32_t id;
int cmd; int32_t cmd;
char data[DATA_BUFFSIZE]; uint8_t data[DATA_BUFFSIZE];
// ignoring string2 atm. // ignoring string2 for now
} rc_packet; } rc_packet;
// =================================== // ===================================
// FUNCTION DEFINITIONS // FUNCTION DEFINITIONS
// =================================== // ===================================
// endianness related functions
bool is_bigendian(void);
int32_t reverse_int32(int32_t n);
// Network related functions // Network related functions
#ifdef _WIN32 #ifdef _WIN32
void net_init_WSA(void); void net_init_WSA(void);
#endif #endif
void net_close(int sd); void net_close(int sd);
int net_connect(const char *host, const char *port); int net_connect(const char *host, const char *port);
int net_send(int sd, const uint8_t *buffer, size_t size); bool net_send_packet(int sd, rc_packet *packet);
int net_send_packet(int sd, rc_packet *packet); rc_packet* net_recv_packet(int sd);
rc_packet* net_recv_packet(int sd);
int net_clean_incoming(int sd, int size);
// Misc stuff // Misc stuff
void usage(void); void usage(void);
#ifndef _WIN32 #ifndef _WIN32
void print_color(int color); void print_color(int color);
#endif #endif
int get_line(char *buffer, int len); int get_line(char *buffer, int len);
int run_terminal_mode(int sock); int run_terminal_mode(int sock);
int run_commands(int argc, char *argv[]); int run_commands(int argc, char *argv[]);
// Rcon protocol related functions // Rcon protocol related functions
rc_packet* packet_build(int id, int cmd, char *s1); rc_packet* packet_build(int id, int cmd, char *s1);
void packet_print(rc_packet *packet); void packet_print(rc_packet *packet);
int rcon_auth(int sock, char *passwd); bool rcon_auth(int sock, char *passwd);
int rcon_command(int sock, char *command); int rcon_command(int sock, char *command);
// ============================================= // =============================================
// GLOBAL VARIABLES // GLOBAL VARIABLES
// ============================================= // =============================================
static int global_raw_output = 0; static int flag_raw_output = 0;
static int global_silent_mode = 0; static int flag_silent_mode = 0;
static int global_print_colors = 1; static int flag_disable_colors = 0;
static int flag_wait_seconds = 0;
static int global_connection_alive = 1; static int global_connection_alive = 1;
static int global_wait_seconds = 0;
static int global_rsock; static int global_rsock;
#ifdef _WIN32 #ifdef _WIN32
@ -125,85 +118,99 @@ void exit_proc(void)
net_close(global_rsock); net_close(global_rsock);
} }
// Check windows & linux behaviour !!! // TODO: check exact windows and linux behaviour
void sighandler(/*int sig*/) void sighandler(int sig)
{ {
if (sig == SIGINT)
putchar('\n');
global_connection_alive = 0; global_connection_alive = 0;
#ifndef _WIN32 #ifndef _WIN32
exit(EXIT_SUCCESS); exit(EXIT_SUCCESS);
#endif #endif
} }
#define MAX_WAIT_TIME 600
unsigned int mcrcon_parse_seconds(char *str)
{
char *end;
long result = strtol(str, &end, 10);
if (errno != 0) {
log_error("-w invalid value.\nerror %d: %s\n", errno, strerror(errno));
exit(EXIT_FAILURE);
}
if (end == str) {
log_error("-w invalid value (not a number?)\n");
exit(EXIT_FAILURE);
}
if (result <= 0 || result > MAX_WAIT_TIME) {
log_error("-w value out of range.\nAcceptable value is 1 - %d (seconds).\n", MAX_WAIT_TIME);
exit(EXIT_FAILURE);
}
return (unsigned int) result;
}
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
int opt;
int terminal_mode = 0; int terminal_mode = 0;
char *host = getenv("MCRCON_HOST"); char *host = getenv("MCRCON_HOST");
char *pass = getenv("MCRCON_PASS"); char *pass = getenv("MCRCON_PASS");
char *port = getenv("MCRCON_PORT"); char *port = getenv("MCRCON_PORT");
if (!port) if (!port) port = "25575";
port = "25575"; if (!host) host = "localhost";
if (!host) // disable output buffering (https://github.com/Tiiffi/mcrcon/pull/39)
host = "localhost"; setvbuf(stdout, NULL, _IONBF, 0);
setvbuf(stderr, NULL, _IONBF, 0);
if(argc < 1 && pass == NULL) if(argc < 1 && pass == NULL) usage();
usage();
// default getopt error handler enabled // default getopt error handler enabled
opterr = 1; opterr = 1;
int opt;
while ((opt = getopt(argc, argv, "vrtcshw:H:p:P:i")) != -1) while ((opt = getopt(argc, argv, "vrtcshw:H:p:P:")) != -1)
{ {
switch (opt) switch (opt) {
{ case 'H': host = optarg; break;
case 'H': host = optarg; break; case 'P': port = optarg; break;
case 'P': port = optarg; break; case 'p': pass = optarg; break;
case 'p': pass = optarg; break; case 'c': flag_disable_colors = 1; break;
case 'C': case 's': flag_silent_mode = 1; break;
case 'c': global_print_colors = 0; break; case 'i': /* reserved for interp mode */ break;
case 'S': case 't': terminal_mode = 1; break;
case 's': global_silent_mode = 1; break; case 'r': flag_raw_output = 1; break;
case 'T':
case 't':
case 'I':
case 'i': terminal_mode = 1; break;
case 'r': global_raw_output = 1; break;
case 'v':
puts(VER_STR"\nhttps://github.com/Tiiffi/mcrcon");
exit(EXIT_SUCCESS);
case 'W':
case 'w': case 'w':
global_wait_seconds = strtol(optarg, NULL, 10); flag_wait_seconds = mcrcon_parse_seconds(optarg);
if (errno != 0) break;
{
fprintf(stderr, "Error %d: %s\n", errno, strerror(errno));
exit(EXIT_FAILURE);
}
break;
case 'h':
case '?': usage(); break;
/*
if(optopt == 'P' || optopt == 'H' || optopt == 'p')
fprintf (stderr, "Option -%c requires an argument.\n\n", optopt);
else fprintf (stderr, "Unknown option -%c\n\n", optopt);
*/
default: exit(EXIT_FAILURE); case 'v':
puts(VER_STR);
puts("Bug reports:\n\ttiiffi+mcrcon at gmail\n\thttps://github.com/Tiiffi/mcrcon/issues/");
exit(EXIT_SUCCESS);
case 'h': usage(); break;
case '?':
default:
puts("Try 'mcrcon -h' or 'man mcrcon' for help.");
exit(EXIT_FAILURE);
} }
} }
if (pass == NULL) if (pass == NULL) {
{ puts("You must give password (-p password).\nTry 'mcrcon -h' or 'man mcrcon' for help.");
fputs("You must give password (-p password). Try 'mcrcon -h' or 'man mcrcon' for help.\n\n", stdout); exit(EXIT_FAILURE);
return 0;
} }
if(optind == argc && terminal_mode == 0) if(optind == argc && terminal_mode == 0) {
terminal_mode = 1; terminal_mode = 1;
}
// safety features to prevent "IO: Connection reset" bug on the server side // safety features to prevent "IO: Connection reset" bug on the server side
atexit(&exit_proc); atexit(&exit_proc);
@ -212,9 +219,10 @@ int main(int argc, char *argv[])
signal(SIGINT, &sighandler); signal(SIGINT, &sighandler);
#ifdef _WIN32 #ifdef _WIN32
net_init_WSA(); net_init_WSA();
console_handle = GetStdHandle(STD_OUTPUT_HANDLE); console_handle = GetStdHandle(STD_OUTPUT_HANDLE);
if (console_handle == INVALID_HANDLE_VALUE) console_handle = NULL; if (console_handle == INVALID_HANDLE_VALUE)
console_handle = NULL;
#endif #endif
// open socket // open socket
@ -223,28 +231,21 @@ int main(int argc, char *argv[])
int exit_code = EXIT_SUCCESS; int exit_code = EXIT_SUCCESS;
// auth & commands // auth & commands
if (rcon_auth(global_rsock, pass)) if (rcon_auth(global_rsock, pass)) {
{ if (terminal_mode) exit_code = run_terminal_mode(global_rsock);
if (terminal_mode) else exit_code = run_commands(argc, argv);
run_terminal_mode(global_rsock);
else
exit_code = run_commands(argc, argv);
} }
else // auth failed else { // auth failed
{ log_error("Authentication failed!\n");
fprintf(stdout, "Authentication failed!\n");
exit_code = EXIT_FAILURE; exit_code = EXIT_FAILURE;
} }
net_close(global_rsock); exit(exit_code);
global_rsock = -1;
return exit_code;
} }
void usage(void) void usage(void)
{ {
fputs( puts(
"Usage: "IN_NAME" [OPTIONS] [COMMANDS]\n\n" "Usage: "IN_NAME" [OPTIONS] [COMMANDS]\n\n"
"Send rcon commands to Minecraft server.\n\n" "Send rcon commands to Minecraft server.\n\n"
"Options:\n" "Options:\n"
@ -255,21 +256,21 @@ void usage(void)
" -s\t\tSilent mode\n" " -s\t\tSilent mode\n"
" -c\t\tDisable colors\n" " -c\t\tDisable colors\n"
" -r\t\tOutput raw packets\n" " -r\t\tOutput raw packets\n"
" -w\t\tWait for specified duration (seconds) between each command (1 - 600s)\n"
" -h\t\tPrint usage\n" " -h\t\tPrint usage\n"
" -v\t\tVersion information\n" " -v\t\tVersion information\n\n"
" -w\t\tWait for specified duration (seconds) between each command\n\n"
"Server address, port and password can be set with following environment variables:\n" "Server address, port and password can be set with following environment variables:\n"
" MCRCON_HOST\n" " MCRCON_HOST\n"
" MCRCON_PORT\n" " MCRCON_PORT\n"
" MCRCON_PASS\n\n" " MCRCON_PASS\n"
,stdout
); );
puts("mcrcon will start in terminal mode if no commands are given."); puts (
puts("Command-line options will override environment variables."); "- mcrcon will start in terminal mode if no commands are given\n"
puts("Rcon commands with spaces must be enclosed in quotes.\n"); "- Command-line options will override environment variables\n"
puts("Example:\n\t"IN_NAME" -H my.minecraft.server -p password \"say Server is restarting!\" save-all stop\n"); "- Rcon commands with spaces must be enclosed in quotes\n"
puts(VER_STR"\nReport bugs to tiiffi+mcrcon at gmail or https://github.com/Tiiffi/mcrcon/issues/\n"); );
puts("Example:\n\t"IN_NAME" -H my.minecraft.server -p password -w 5 \"say Server is restarting!\" save-all stop\n");
#ifdef _WIN32 #ifdef _WIN32
puts("Press enter to exit."); puts("Press enter to exit.");
@ -289,9 +290,8 @@ void net_init_WSA(void)
WORD version = MAKEWORD(2, 2); WORD version = MAKEWORD(2, 2);
int err = WSAStartup(version, &wsadata); int err = WSAStartup(version, &wsadata);
if (err != 0) if (err != 0) {
{ log_error("WSAStartup failed. Error: %d.\n", err);
fprintf(stderr, "WSAStartup failed. Error: %d.\n", err);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
} }
@ -300,15 +300,14 @@ void net_init_WSA(void)
// socket close and cleanup // socket close and cleanup
void net_close(int sd) void net_close(int sd)
{ {
#ifdef _WIN32 #ifdef _WIN32
closesocket(sd); closesocket(sd);
WSACleanup(); WSACleanup();
#else #else
close(sd); close(sd);
#endif #endif
} }
// Opens and connects socket
// http://man7.org/linux/man-pages/man3/getaddrinfo.3.html // http://man7.org/linux/man-pages/man3/getaddrinfo.3.html
// https://bugs.chromium.org/p/chromium/issues/detail?id=44489 // https://bugs.chromium.org/p/chromium/issues/detail?id=44489
int net_connect(const char *host, const char *port) int net_connect(const char *host, const char *port)
@ -328,27 +327,26 @@ int net_connect(const char *host, const char *port)
#endif #endif
int ret = getaddrinfo(host, port, &hints, &server_info); int ret = getaddrinfo(host, port, &hints, &server_info);
if (ret != 0) if (ret != 0) {
{ log_error("Name resolution failed.\n");
fprintf(stderr, "Name resolution failed.\n");
#ifdef _WIN32 #ifdef _WIN32
fprintf(stderr, "Error %d: %s", ret, gai_strerror(ret)); log_error("Error %d: %s", ret, gai_strerror(ret));
#else #else
fprintf(stderr, "Error %d: %s\n", ret, gai_strerror(ret)); log_error("Error %d: %s\n", ret, gai_strerror(ret));
#endif #endif
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
// Go through the hosts and try to connect // Go through the hosts and try to connect
for (p = server_info; p != NULL; p = p->ai_next) for (p = server_info; p != NULL; p = p->ai_next) {
{
sd = socket(p->ai_family, p->ai_socktype, p->ai_protocol); sd = socket(p->ai_family, p->ai_socktype, p->ai_protocol);
if (sd == -1) if (sd == -1)
continue; continue;
ret = connect(sd, p->ai_addr, p->ai_addrlen); ret = connect(sd, p->ai_addr, p->ai_addrlen);
if (ret == -1) if (ret == -1) {
{
net_close(sd); net_close(sd);
continue; continue;
} }
@ -356,102 +354,74 @@ int net_connect(const char *host, const char *port)
break; break;
} }
if (p == NULL) if (p == NULL) {
{
/* TODO (Tiiffi): Check why windows does not report errors */ /* TODO (Tiiffi): Check why windows does not report errors */
fprintf(stderr, "Connection failed.\n"); log_error("Connection failed.\n");
#ifndef _WIN32 #ifndef _WIN32
fprintf(stderr, "Error %d: %s\n", errno, strerror(errno)); log_error("Error %d: %s\n", errno, strerror(errno));
#endif #endif
freeaddrinfo(server_info); freeaddrinfo(server_info);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
freeaddrinfo(server_info); freeaddrinfo(server_info);
return sd; return sd;
} }
int net_send(int sd, const uint8_t *buff, size_t size) bool net_send_packet(int sd, rc_packet *packet)
{ {
size_t sent = 0; size_t sent = 0;
size_t size = packet->size + sizeof(int32_t);
size_t left = size; size_t left = size;
while (sent < size) char *p = (char *) packet;
{
int result = send(sd, (const char *) buff + sent, left, 0);
if (result == -1) return -1; while (sent < size) {
ssize_t result = send(sd, p + sent, left, 0);
if (result == -1) return false;
sent += result; sent += result;
left -= sent; left -= sent;
} }
return 0; return true;
}
int net_send_packet(int sd, rc_packet *packet)
{
int len;
int total = 0; // bytes we've sent
int bytesleft; // bytes left to send
int ret = -1;
bytesleft = len = packet->size + sizeof(int);
while(total < len)
{
ret = send(sd, (char *) packet + total, bytesleft, 0);
if(ret == -1) { break; }
total += ret;
bytesleft -= ret;
}
return ret == -1 ? -1 : 1;
} }
rc_packet *net_recv_packet(int sd) rc_packet *net_recv_packet(int sd)
{ {
int psize; int32_t psize;
static rc_packet packet = {0, 0, 0, { 0x00 }}; static rc_packet packet = {0};
// packet.size = packet.id = packet.cmd = 0; ssize_t ret = recv(sd, (char *) &psize, sizeof(psize), 0);
int ret = recv(sd, (char *) &psize, sizeof(int), 0); if (ret == 0) {
log_error("Connection lost.\n");
if (ret == 0)
{
fprintf(stderr, "Connection lost.\n");
global_connection_alive = 0; global_connection_alive = 0;
return NULL; return NULL;
} }
if (ret != sizeof(int)) if (ret != sizeof(psize)) {
{ log_error("Error: recv() failed.\n");
fprintf(stderr, "Error: recv() failed. Invalid packet size (%d).\n", ret);
global_connection_alive = 0; global_connection_alive = 0;
return NULL; return NULL;
} }
if (psize < 10 || psize > DATA_BUFFSIZE) if (psize < MIN_PACKET_SIZE || psize > MAX_PACKET_SIZE) {
{ log_error("Error: Invalid packet size (%d).\n", psize);
fprintf(stderr, "Warning: invalid packet size (%d). Must over 10 and less than %d.\n", psize, DATA_BUFFSIZE); global_connection_alive = 0;
if(psize > DATA_BUFFSIZE || psize < 0) psize = DATA_BUFFSIZE;
net_clean_incoming(sd, psize);
return NULL; return NULL;
} }
packet.size = psize; packet.size = psize;
char *p = (char *) &packet;
int received = 0; int received = 0;
while (received < psize) while (received < psize) {
{ ret = recv(sd, p + sizeof(int32_t) + received, psize - received, 0);
ret = recv(sd, (char *) &packet + sizeof(int) + received, psize - received, 0); if (ret == 0) {
if (ret == 0) /* connection closed before completing receving */ log_error("Connection lost.\n");
{
fprintf(stderr, "Connection lost.\n");
global_connection_alive = 0; global_connection_alive = 0;
return NULL; return NULL;
} }
@ -462,27 +432,11 @@ rc_packet *net_recv_packet(int sd)
return &packet; return &packet;
} }
int net_clean_incoming(int sd, int size)
{
char tmp[size];
int ret = recv(sd, tmp, size, 0);
if(ret == 0)
{
fprintf(stderr, "Connection lost.\n");
global_connection_alive = 0;
}
return ret;
}
void print_color(int color) void print_color(int color)
{ {
// sh compatible color array // sh compatible color array
#ifndef _WIN32 #ifndef _WIN32
char *colors[] = char *colors[] = {
{
"\033[0;30m", /* 00 BLACK 0x30 */ "\033[0;30m", /* 00 BLACK 0x30 */
"\033[0;34m", /* 01 BLUE 0x31 */ "\033[0;34m", /* 01 BLUE 0x31 */
"\033[0;32m", /* 02 GREEN 0x32 */ "\033[0;32m", /* 02 GREEN 0x32 */
@ -494,7 +448,7 @@ void print_color(int color)
"\033[0;1;30m", /* 08 DGREY 0x38 */ "\033[0;1;30m", /* 08 DGREY 0x38 */
"\033[0;1;34m", /* 09 LBLUE 0x39 */ "\033[0;1;34m", /* 09 LBLUE 0x39 */
"\033[0;1;32m", /* 10 LGREEN 0x61 */ "\033[0;1;32m", /* 10 LGREEN 0x61 */
"\033[0:1;36m", /* 11 LCYAN 0x62 */ "\033[0;1;36m", /* 11 LCYAN 0x62 */
"\033[0;1;31m", /* 12 LRED 0x63 */ "\033[0;1;31m", /* 12 LRED 0x63 */
"\033[0;1;35m", /* 13 LPURPLE 0x64 */ "\033[0;1;35m", /* 13 LPURPLE 0x64 */
"\033[0;1;33m", /* 14 YELLOW 0x65 */ "\033[0;1;33m", /* 14 YELLOW 0x65 */
@ -502,22 +456,23 @@ void print_color(int color)
"\033[4m" /* 16 UNDERLINE 0x6e */ "\033[4m" /* 16 UNDERLINE 0x6e */
}; };
if(color == 0 || color == 0x72) /* 0x72: 'r' */ if (color == 0 || color == 0x72) {
{ fputs("\033[0m", stdout); // cancel color
fputs("\033[0m", stdout); /* CANCEL COLOR */
} }
else else
#endif #endif
{ {
if(color >= 0x61 && color <= 0x66) color -= 0x57; if (color >= 0x61 && color <= 0x66) color -= 0x57;
else if(color >= 0x30 && color <= 0x39) color -= 0x30; else if (color >= 0x30 && color <= 0x39)
else if(color == 0x6e) color=16; /* 0x6e: 'n' */ color -= 0x30;
else if (color == 0x6e)
color = 16;
else return; else return;
#ifndef _WIN32 #ifndef _WIN32
fputs(colors[color], stdout); fputs(colors[color], stdout);
#else #else
SetConsoleTextAttribute(console_handle, color); SetConsoleTextAttribute(console_handle, color);
#endif #endif
} }
} }
@ -525,124 +480,134 @@ void print_color(int color)
// this hacky mess might use some optimizing // this hacky mess might use some optimizing
void packet_print(rc_packet *packet) void packet_print(rc_packet *packet)
{ {
if (global_raw_output == 1) uint8_t *data = packet->data;
{
for (int i = 0; packet->data[i] != 0; ++i) putchar(packet->data[i]); if (flag_raw_output == 1) {
for (int i = 0; data[i] != 0; ++i) {
putchar(data[i]);
}
return; return;
} }
int i; int i;
int def_color = 0; int default_color = 0;
#ifdef _WIN32 #ifdef _WIN32
CONSOLE_SCREEN_BUFFER_INFO console_info; CONSOLE_SCREEN_BUFFER_INFO console_info;
if (GetConsoleScreenBufferInfo(console_handle, &console_info) != 0) {
if (GetConsoleScreenBufferInfo(console_handle, &console_info) != 0) default_color = console_info.wAttributes + 0x30;
def_color = console_info.wAttributes + 0x30; } else default_color = 0x37;
else
def_color = 0x37;
#endif #endif
// colors enabled so try to handle the bukkit colors for terminal // colors enabled so try to handle the bukkit colors for terminal
if (global_print_colors == 1) if (flag_disable_colors == 0) {
{ for (i = 0; data[i] != 0; ++i) {
for (i = 0; (unsigned char) packet->data[i] != 0; ++i) if (data[i] == 0x0A) print_color(default_color);
{ else if(data[i] == 0xc2 && data[i + 1] == 0xa7) {
if (packet->data[i] == 0x0A) print_color(def_color); i += 2;
else if((unsigned char) packet->data[i] == 0xc2 && (unsigned char) packet->data[i+1] == 0xa7){ print_color(data[i]);
print_color(packet->data[i+=2]);
continue; continue;
} }
putchar(packet->data[i]); putchar(data[i]);
} }
print_color(def_color); // cancel coloring print_color(default_color); // cancel coloring
} }
// strip colors // strip colors
else else {
{ for (i = 0; data[i] != 0; ++i) {
for (i = 0; (unsigned char) packet->data[i] != 0; ++i) if (data[i] == 0xc2 && data[i + 1] == 0xa7) {
{ i += 2;
if ((unsigned char) packet->data[i] == 0xc2 && (unsigned char) packet->data[i+1] == 0xa7){
i+=2;
continue; continue;
} }
putchar(packet->data[i]); putchar(data[i]);
} }
} }
// print newline if string has no newline // print newline if string has no newline
if (packet->data[i-1] != 10 && packet->data[i-1] != 13) if (data[i - 1] != 10 && data[i - 1] != 13) {
putchar('\n'); putchar('\n');
}
} }
rc_packet *packet_build(int id, int cmd, char *s1) rc_packet *packet_build(int id, int cmd, char *s)
{ {
static rc_packet packet = {0, 0, 0, { 0x00 }}; static rc_packet packet = {0};
// size + id + cmd + s1 + s2 NULL terminator // NOTE(Tiiffi): Issue report states that maximum command packet size is 1460 bytes:
int s1_len = strlen(s1); // https://github.com/Tiiffi/mcrcon/issues/45#issuecomment-1000940814
if (s1_len > DATA_BUFFSIZE) // https://mctools.readthedocs.io/en/master/rcon.html
{ // Have to do some testing to confirm!
fprintf(stderr, "Warning: Command string too long (%d). Maximum allowed: %d.\n", s1_len, DATA_BUFFSIZE);
int len = strlen(s);
if (len > MAX_COMMAND_LENGTH) {
log_error("Warning: Command string too long (%d). Maximum allowed: %d.\n", len, MAX_COMMAND_LENGTH);
return NULL; return NULL;
} }
packet.size = sizeof(int) * 2 + s1_len + 2; packet.size = sizeof packet.id + sizeof packet.cmd + len + 2;
packet.id = id; packet.id = id;
packet.cmd = cmd; packet.cmd = cmd;
strncpy(packet.data, s1, DATA_BUFFSIZE); memcpy(packet.data, s, len);
packet.data[len] = 0;
packet.data[len + 1] = 0;
return &packet; return &packet;
} }
int rcon_auth(int sock, char *passwd) bool rcon_auth(int sock, char *passwd)
{ {
int ret;
rc_packet *packet = packet_build(RCON_PID, RCON_AUTHENTICATE, passwd); rc_packet *packet = packet_build(RCON_PID, RCON_AUTHENTICATE, passwd);
if (packet == NULL) if (packet == NULL)
return 0; return 0;
ret = net_send_packet(sock, packet); if (!net_send_packet(sock, packet)) {
if (!ret)
return 0; // send failed return 0; // send failed
}
receive:
packet = net_recv_packet(sock); packet = net_recv_packet(sock);
if (packet == NULL) if (packet == NULL)
return 0; return 0;
// return 1 if authentication OK /* Valve rcon sends empty "RCON_RESPONSEVALUE" packet before real auth response
return packet->id == -1 ? 0 : 1; * so we have to check packet type and try again if necessary.
*/
if (packet->cmd != RCON_AUTH_RESPONSE) {
goto receive;
}
// return true if authentication OK
return packet->id == -1 ? false : true;
} }
// TODO: Add proper error handling and reporting!
int rcon_command(int sock, char *command) int rcon_command(int sock, char *command)
{ {
rc_packet *packet = packet_build(RCON_PID, RCON_EXEC_COMMAND, command); rc_packet *packet = packet_build(RCON_PID, RCON_EXEC_COMMAND, command);
if (packet == NULL) if (packet == NULL) {
{ log_error("Error: packet build() failed!\n");
global_connection_alive = 0;
return 0; return 0;
} }
net_send_packet(sock, packet); if (!net_send_packet(sock, packet)) {
log_error("Error: net_send_packet() failed!\n");
return 0;
}
packet = net_recv_packet(sock); packet = net_recv_packet(sock);
if (packet == NULL) if (packet == NULL) {
log_error("Error: net_recv_packet() failed!\n");
return 0; return 0;
if (packet->id != RCON_PID)
return 0;
if (!global_silent_mode)
{
/*
if(packet->size == 10) {
printf("Unknown command \"%s\". Type \"help\" or \"?\" for help.\n", command);
} }
else
*/ if (packet->id != RCON_PID) {
if (packet->size > 10) log_error("Error: invalid packet id!\n");
packet_print(packet); return 0;
}
if (!flag_silent_mode) {
if (packet->size > 10)
packet_print(packet);
} }
return 1; return 1;
@ -652,89 +617,84 @@ int run_commands(int argc, char *argv[])
{ {
int i = optind; int i = optind;
for (;;) for (;;) {
{
if (!rcon_command(global_rsock, argv[i])) if (!rcon_command(global_rsock, argv[i]))
return EXIT_FAILURE; return EXIT_FAILURE;
if (++i >= argc) i++;
if (i >= argc)
return EXIT_SUCCESS; return EXIT_SUCCESS;
if (global_wait_seconds > 0) if (flag_wait_seconds > 0) {
{
#ifdef _WIN32 #ifdef _WIN32
Sleep(global_wait_seconds * 1000); Sleep(flag_wait_seconds * 1000);
#else #else
sleep(global_wait_seconds); sleep(flag_wait_seconds);
#endif #endif
} }
} }
} }
// interactive terminal mode // terminal mode
int run_terminal_mode(int sock) int run_terminal_mode(int sock)
{ {
int ret = 0; char command[MAX_COMMAND_LENGTH] = {0};
char command[DATA_BUFFSIZE] = {0x00};
puts("Logged in. Type \"Q\" to quit!"); puts("Logged in.\nType 'Q' or press Ctrl-D / Ctrl-C to disconnect.");
while (global_connection_alive) while (global_connection_alive) {
{ putchar('>');
int len = get_line(command, DATA_BUFFSIZE);
if(command[0] == 'Q' && command[1] == 0) int len = get_line(command, MAX_COMMAND_LENGTH);
if (len < 1) continue;
if (strcasecmp(command, "Q") == 0) break;
if (len > 0 && global_connection_alive) {
if (!rcon_command(sock, command)) {
return EXIT_FAILURE;
}
}
/* Special case for "stop" command to prevent server-side bug.
* https://bugs.mojang.com/browse/MC-154617
*
* NOTE: This is hacky workaround which should be handled better to
* ensure compatibility with other servers using source RCON.
* NOTE: strcasecmp() is POSIX function.
*/
if (strcasecmp(command, "stop") == 0) {
break; break;
}
if(len > 0 && global_connection_alive)
ret += rcon_command(sock, command);
command[0] = len = 0;
} }
return ret; return EXIT_SUCCESS;
} }
// gets line from stdin and deals with rubbish left in the input buffer // gets line from stdin and deals with rubbish left in the input buffer
int get_line(char *buffer, int bsize) int get_line(char *buffer, int bsize)
{ {
int ch, len;
fputs(">", stdout);
char *ret = fgets(buffer, bsize, stdin); char *ret = fgets(buffer, bsize, stdin);
if (ret == NULL) exit(EXIT_FAILURE); if (ret == NULL) {
if (ferror(stdin)) {
if (buffer[0] == 0) global_connection_alive = 0; log_error("Error %d: %s\n", errno, strerror(errno));
exit(EXIT_FAILURE);
}
putchar('\n');
exit(EXIT_SUCCESS);
}
// remove unwanted characters from the buffer // remove unwanted characters from the buffer
buffer[strcspn(buffer, "\r\n")] = '\0'; buffer[strcspn(buffer, "\r\n")] = '\0';
len = strlen(buffer); int len = strlen(buffer);
// clean input buffer if needed // clean input buffer if needed
if (len == bsize - 1) if (len == bsize - 1) {
int ch;
while ((ch = getchar()) != '\n' && ch != EOF); while ((ch = getchar()) != '\n' && ch != EOF);
}
return len; return len;
} }
// http://www.ibm.com/developerworks/aix/library/au-endianc/
bool is_bigendian(void)
{
const int32_t n = 1;
if (*(uint8_t *) &n == 0 ) return true;
return false;
}
int32_t reverse_int32(int32_t n)
{
int32_t tmp;
uint8_t *t = (uint8_t *) &tmp;
uint8_t *p = (uint8_t *) &n;
t[0] = p[3];
t[1] = p[2];
t[2] = p[1];
t[3] = p[0];
return tmp;
}