2019-07-07 18:56:21 +02:00
[![GitHub issues ](https://img.shields.io/github/issues/itzg/mc-router.svg )](https://github.com/itzg/mc-router/issues)
[![Docker Pulls ](https://img.shields.io/docker/pulls/itzg/mc-router.svg )](https://cloud.docker.com/u/itzg/repository/docker/itzg/mc-router)
2021-11-14 20:23:42 +01:00
[![test ](https://github.com/itzg/mc-router/actions/workflows/test.yml/badge.svg )](https://github.com/itzg/mc-router/actions/workflows/test.yml)
2019-07-07 18:56:21 +02:00
[![GitHub release ](https://img.shields.io/github/release/itzg/mc-router.svg )](https://github.com/itzg/mc-router/releases)
2021-11-14 20:23:42 +01:00
[![Discord ](https://img.shields.io/discord/660567679458869252?label=discord )](https://discord.gg/JK2v3rJ9ec)
2019-07-07 18:56:21 +02:00
[![Buy me a coffee ](https://img.shields.io/badge/Donate-Buy%20me%20a%20coffee-orange.svg )](https://www.buymeacoffee.com/itzg)
2018-05-08 05:16:01 +02:00
Routes Minecraft client connections to backend servers based upon the requested server address.
2023-09-09 19:06:59 +02:00
## Usage
2018-05-08 05:16:01 +02:00
```text
2020-01-04 21:22:01 +01:00
-api-binding host:port
2024-12-19 14:37:08 +01:00
The host:port bound for servicing API requests (env API_BINDING)
2022-01-07 00:41:10 +01:00
-auto-scale-up
2024-12-19 14:37:08 +01:00
Increase Kubernetes StatefulSet Replicas (only) from 0 to 1 on respective backend servers when accessed (env AUTO_SCALE_UP)
-clients-to-allow value
Zero or more client IP addresses or CIDRs to allow. Takes precedence over deny. (env CLIENTS_TO_ALLOW)
-clients-to-deny value
Zero or more client IP addresses or CIDRs to deny. Ignored if any configured to allow (env CLIENTS_TO_DENY)
2020-01-04 21:22:01 +01:00
-connection-rate-limit int
2024-12-19 14:37:08 +01:00
Max number of connections to allow per second (env CONNECTION_RATE_LIMIT) (default 1)
2020-01-04 21:22:01 +01:00
-cpu-profile string
2024-12-19 14:37:08 +01:00
Enables CPU profiling and writes to given path (env CPU_PROFILE)
2020-01-04 21:22:01 +01:00
-debug
2024-12-19 14:37:08 +01:00
Enable debug logs (env DEBUG)
2023-09-09 19:06:59 +02:00
-default string
2024-12-19 14:37:08 +01:00
host:port of a default Minecraft server to use when mapping not found (env DEFAULT)
2023-09-09 19:06:59 +02:00
-docker-refresh-interval int
2024-12-19 14:37:08 +01:00
Refresh interval in seconds for the Docker integrations (env DOCKER_REFRESH_INTERVAL) (default 15)
-docker-socket string
Path to Docker socket to use (env DOCKER_SOCKET) (default "unix:///var/run/docker.sock")
2023-09-09 19:06:59 +02:00
-docker-timeout int
2024-12-19 14:37:08 +01:00
Timeout configuration in seconds for the Docker integrations (env DOCKER_TIMEOUT)
2024-08-05 00:22:39 +02:00
-in-docker
2024-12-19 14:37:08 +01:00
Use Docker service discovery (env IN_DOCKER)
2023-09-09 19:06:59 +02:00
-in-docker-swarm
2024-12-19 14:37:08 +01:00
Use Docker Swarm service discovery (env IN_DOCKER_SWARM)
2020-01-04 21:22:01 +01:00
-in-kube-cluster
2024-12-19 14:37:08 +01:00
Use in-cluster Kubernetes config (env IN_KUBE_CLUSTER)
2020-01-04 21:22:01 +01:00
-kube-config string
2024-12-19 14:37:08 +01:00
The path to a Kubernetes configuration file (env KUBE_CONFIG)
2023-09-09 19:06:59 +02:00
-mapping value
2024-12-19 14:37:08 +01:00
Comma or newline delimited or repeated mappings of externalHostname=host:port (env MAPPING)
2020-01-04 21:22:01 +01:00
-metrics-backend string
2024-12-19 14:37:08 +01:00
Backend to use for metrics exposure/publishing: discard,expvar,influxdb (env METRICS_BACKEND) (default "discard")
2020-01-04 21:22:01 +01:00
-metrics-backend-config-influxdb-addr string
2024-12-19 14:37:08 +01:00
(env METRICS_BACKEND_CONFIG_INFLUXDB_ADDR)
2020-01-04 21:22:01 +01:00
-metrics-backend-config-influxdb-database string
2024-12-19 14:37:08 +01:00
(env METRICS_BACKEND_CONFIG_INFLUXDB_DATABASE)
2020-01-04 21:22:01 +01:00
-metrics-backend-config-influxdb-interval duration
2024-12-19 14:37:08 +01:00
(env METRICS_BACKEND_CONFIG_INFLUXDB_INTERVAL) (default 1m0s)
2020-01-04 21:22:01 +01:00
-metrics-backend-config-influxdb-password string
2024-12-19 14:37:08 +01:00
(env METRICS_BACKEND_CONFIG_INFLUXDB_PASSWORD)
2020-01-04 21:22:01 +01:00
-metrics-backend-config-influxdb-retention-policy string
2024-12-19 14:37:08 +01:00
(env METRICS_BACKEND_CONFIG_INFLUXDB_RETENTION_POLICY)
2020-01-04 21:22:01 +01:00
-metrics-backend-config-influxdb-tags value
2024-12-19 14:37:08 +01:00
any extra tags to be included with all reported metrics (env METRICS_BACKEND_CONFIG_INFLUXDB_TAGS)
2020-01-04 21:22:01 +01:00
-metrics-backend-config-influxdb-username string
2024-12-19 14:37:08 +01:00
(env METRICS_BACKEND_CONFIG_INFLUXDB_USERNAME)
2023-09-09 19:06:59 +02:00
-ngrok-token string
2024-12-19 14:37:08 +01:00
If set, an ngrok tunnel will be established. It is HIGHLY recommended to pass as an environment variable. (env NGROK_TOKEN)
2020-01-04 21:22:01 +01:00
-port port
2024-12-19 14:37:08 +01:00
The port bound to listen for Minecraft client connections (env PORT) (default 25565)
2024-07-07 18:13:12 +02:00
-receive-proxy-protocol
2024-12-19 14:37:08 +01:00
Receive PROXY protocol from backend servers, by default trusts every proxy header that it receives, combine with -trusted-proxies to specify a list of trusted proxies (env RECEIVE_PROXY_PROTOCOL)
2022-12-05 14:58:13 +01:00
-routes-config string
2024-12-19 14:37:08 +01:00
Name or full path to routes config file (env ROUTES_CONFIG)
2023-09-09 19:06:59 +02:00
-simplify-srv
2024-12-19 14:37:08 +01:00
Simplify fully qualified SRV records for mapping (env SIMPLIFY_SRV)
2024-07-07 18:13:12 +02:00
-trusted-proxies value
2024-12-19 14:37:08 +01:00
Comma delimited list of CIDR notation IP blocks to trust when receiving PROXY protocol (env TRUSTED_PROXIES)
2023-09-09 19:06:59 +02:00
-use-proxy-protocol
2024-12-19 14:37:08 +01:00
Send PROXY protocol to backend servers (env USE_PROXY_PROTOCOL)
2020-01-04 21:22:01 +01:00
-version
2024-12-19 14:37:08 +01:00
Output version and exit (env VERSION)
2018-05-08 05:16:01 +02:00
```
2021-08-10 23:53:51 +02:00
2023-09-09 19:06:59 +02:00
## Docker Multi-Architecture Image
2020-12-29 17:50:43 +01:00
The [multi-architecture image published at Docker Hub ](https://hub.docker.com/repository/docker/itzg/mc-router ) supports amd64, arm64, and arm32v6 (i.e. RaspberryPi).
2023-09-09 19:06:59 +02:00
## Docker Compose Usage
2019-07-27 18:36:16 +02:00
2023-10-28 23:54:46 +02:00
The diagram below shows how this `docker-compose.yml` configures two Minecraft server services named `vanilla` and `forge` , which also become the internal network aliases. _Notice those services don't need their ports exposed since the internal networking allows for the inter-container access._
2019-07-27 18:36:16 +02:00
2023-10-28 23:54:46 +02:00
```yaml
version: "3.8"
services:
vanilla:
image: itzg/minecraft-server
environment:
EULA: "TRUE"
forge:
image: itzg/minecraft-server
environment:
EULA: "TRUE"
TYPE: FORGE
router:
image: ${MC_ROUTER_IMAGE:-itzg/mc-router}
depends_on:
- forge
- vanilla
environment:
MAPPING: |
vanilla.example.com=vanilla:25565
forge.example.com=forge:25565
ports:
- "25565:25565"
```
The `router` service is only one of the services that needs to exposed on the external network. The `MAPPING` declares how the hostname users will enter into their Minecraft client will map to the internal services.
2019-07-27 18:36:16 +02:00
![](docs/compose-diagram.png)
2022-01-07 00:41:10 +01:00
2023-10-28 23:54:46 +02:00
To test out this example, add these two entries to my "hosts" file:
2019-07-27 18:36:16 +02:00
```
127.0.0.1 vanilla.example.com
127.0.0.1 forge.example.com
```
2022-01-07 00:41:10 +01:00
2024-08-05 00:22:39 +02:00
### Using Docker auto-discovery
When running `mc-router` in a Docker environment you can pass the `--in-docker` or `--in-docker-swarm`
command-line argument and it will poll the Docker API periodically to find all the running
containers/services for Minecraft instances. To enable discovery you have to set the `mc-router.host`
label on the container. These are the labels scanned:
- `mc-router.host` : Used to configure the hostname the Minecraft clients would use to
connect to the server. The container/service endpoint will be used as the routed backend. You can
use more than one hostname by splitting it with a comma.
- `mc-router.port` : This value must be set to the port the Minecraft server is listening on.
The default value is 25565.
- `mc-router.default` : Set this to a truthy value to make this server the default backend.
Please note that `mc-router.host` is still required to be set.
- `mc-router.network` : Specify the network you are using for the router if multiple are
present in the container/service. You can either use the network ID, it's full name or an alias.
#### Example Docker deployment
Refer to [this example docker-compose.yml ](docs/sd-docker.docker-compose.yml ) to see how to
configure two different Minecraft servers and a `mc-router` instance for use with Docker service discovery.
#### Example Docker Swarm deployment
Refer to [this example docker-compose.yml ](docs/swarm.docker-compose.yml ) to see how to
configure two different Minecraft servers and a `mc-router` instance for use with Docker Swarm service discovery.
2023-09-09 19:06:59 +02:00
## Routing Configuration
2023-05-24 15:24:39 +02:00
The routing configuration allows routing via a config file rather than a command.
You need to set `-routes-config` or `ROUTES_CONFIG` env variable.
The following shows a JSON file for routes config, where `default-server` can also be `null` or omitted:
```json
{
"default-server": "vanilla:25565",
"mappings": {
"vanilla.example.com": "vanilla:25565",
"forge.example.com": "forge:25565"
}
}
```
2023-09-09 19:06:59 +02:00
## Kubernetes Usage
2019-07-27 18:36:16 +02:00
2023-09-09 19:06:59 +02:00
### Using Kubernetes Service auto-discovery
2018-05-27 21:53:22 +02:00
2022-10-24 05:18:10 +02:00
When running `mc-router` as a Kubernetes Pod and you pass the `--in-kube-cluster` command-line argument, then it will automatically watch for any services annotated with
- `mc-router.itzg.me/externalServerName` : The value of the annotation will be registered as the external hostname Minecraft clients would used to connect to the routed service. The service's clusterIP and target port are used as the routed backend. You can use more hostnames by splitting them with comma.
- `mc-router.itzg.me/defaultServer` : The service's clusterIP and target port are used as the default if no other `externalServiceName` annotations applies.
2018-05-27 21:53:22 +02:00
For example, start `mc-router` 's container spec with
```yaml
image: itzg/mc-router
name: mc-router
args: ["--in-kube-cluster"]
```
and configure the backend minecraft server's service with the annotation:
```yaml
apiVersion: v1
kind: Service
metadata:
name: mc-forge
annotations:
"mc-router.itzg.me/externalServerName": "external.host.name"
```
2021-12-04 16:25:28 +01:00
you can use multiple host names:
```yaml
apiVersion: v1
kind: Service
metadata:
name: mc-forge
annotations:
"mc-router.itzg.me/externalServerName": "external.host.name,other.host.name"
```
2022-10-24 05:18:10 +02:00
mc-router will pick the service port named either `minecraft` or `mc-router` . If neither port names exist, it will use port value 25565.
2023-09-09 19:06:59 +02:00
### Example Kubernetes deployment
2018-05-25 03:19:14 +02:00
2022-01-07 00:41:10 +01:00
[This example deployment ](docs/k8s-example-auto.yaml )
2018-05-26 21:03:04 +02:00
* Declares an `mc-router` service that exposes a node port 25565
* Declares a service account with access to watch and list services
* Declares `--in-kube-cluster` in the `mc-router` container arguments
2022-01-07 00:41:10 +01:00
* Two "backend" Minecraft servers are declared each with an
2021-12-04 16:25:28 +01:00
`"mc-router.itzg.me/externalServerName"` annotation that declares their external server name(s)
2018-05-25 03:19:14 +02:00
```bash
2018-05-26 21:03:04 +02:00
kubectl apply -f https://raw.githubusercontent.com/itzg/mc-router/master/docs/k8s-example-auto.yaml
2018-05-25 03:19:14 +02:00
```
2018-05-26 21:03:04 +02:00
![](docs/example-deployment-auto.drawio.png)
2018-05-25 03:19:14 +02:00
2023-09-09 19:06:59 +02:00
##### Notes
2018-05-25 03:26:33 +02:00
* This deployment assumes two persistent volume claims: `mc-stable` and `mc-snapshot`
2022-01-07 00:41:10 +01:00
* I extended the allowed node port range by adding `--service-node-port-range=25000-32767`
to `/etc/kubernetes/manifests/kube-apiserver.yaml`
2023-09-09 19:06:59 +02:00
##### Auto Scale Up
2022-01-07 00:41:10 +01:00
The `-auto-scale-up` flag argument makes the router "wake up" any stopped backend servers, by changing `replicas: 0` to `replicas: 1` .
This requires using `kind: StatefulSet` instead of `kind: Service` for the Minecraft backend servers.
It also requires the `ClusterRole` to permit `get` + `update` for `statefulsets` & `statefulsets/scale` ,
e.g. like this (or some equivalent more fine-grained one to only watch/list services+statefulsets, and only get+update scale):
```yaml
apiVersion: rbac.authorization.k8s.io/v1
kind: ClusterRole
metadata:
name: services-watcher
rules:
- apiGroups: [""]
resources: ["services"]
verbs: ["watch","list"]
- apiGroups: ["apps"]
resources: ["statefulsets", "statefulsets/scale"]
verbs: ["watch","list","get","update"]
```
2019-07-15 00:02:12 +02:00
2024-08-06 19:35:57 +02:00
Make sure to set `StatefulSet.metadata.name` and `StatefulSet.spec.serviceName` to the same value;
otherwise, autoscaling will not trigger:
```yaml
apiVersion: v1
kind: Service
metadata:
name: mc-forge
spec:
type: ClusterIP
annotations:
"mc-router.itzg.me/defaultServer": "true"
"mc-router.itzg.me/externalServerName": "external.host.name"
---
apiVersion: apps/v1
kind: StatefulSet
metadata:
name: mc-forge
spec:
serviceName: mc-forge
```
2023-09-09 19:06:59 +02:00
## REST API
* `GET /routes` (with `Accept: application/json` )
Retrieves the currently configured routes
* `POST /routes` (with `Content-Type: application/json` )
Registers a route given a JSON body structured like:
```json
{
"serverAddress": "CLIENT REQUESTED SERVER ADDRESS",
"backend": "HOST:PORT"
}
```
* `POST /defaultRoute` (with `Content-Type: application/json` )
Registers a default route to the given backend. JSON body is structured as:
```json
{
"backend": "HOST:PORT"
}
```
* `DELETE /routes/{serverAddress}`
Deletes an existing route for the given `serverAddress`
## ngrok
mc-router has built-in support to run as an [ngrok agent ](https://ngrok.com/docs/secure-tunnels/ngrok-agent/ ). To enable this support, pass [an ngrok authtoken ](https://ngrok.com/docs/secure-tunnels/ngrok-agent/tunnel-authtokens/#per-agent-authtokens ) to the command-line argument or environment variable, [shown above ](#usage ).
### Ngrok Quick Start
Create/access an ngrok account and [allocate an agent authtoken from the dashboard ](https://dashboard.ngrok.com/tunnels/authtokens ).
In a new directory, create a file called `.env` with the allocated token
```dotenv
NGROK_TOKEN=...
```
In the same directory, create the following compose file:
```yaml
version: "3.8"
services:
mc:
image: itzg/minecraft-server
environment:
EULA: true
volumes:
- mc-data:/data
# No port mapping since mc-router connects over compose network
router:
image: itzg/mc-router
environment:
DEFAULT: mc:25565
NGROK_TOKEN: ${NGROK_TOKEN}
# No port mapping needed since it routes through ngrok tunnel
volumes:
mc-data: {}
```
Start the compose project:
```shell
docker compose up -d
```
Grab the mc-router logs using:
```shell
docker compose logs router
```
From those logs, locate the `ngrokUrl` parameter from the "Listening" info log message, such as `tcp://8.tcp.ngrok.io:99999` .
In the Minecraft client, the server address will be the part after the "tcp://" prefix, such as `8.tcp.ngrok.io:99999` .
## Development
2019-07-15 00:02:12 +02:00
2023-09-09 19:06:59 +02:00
### Building locally with Docker
2019-07-15 00:02:12 +02:00
```bash
2021-12-05 01:31:18 +01:00
docker build -t mc-router .
2019-07-15 00:02:12 +02:00
```
2023-09-09 19:06:59 +02:00
### Build locally without Docker
2022-01-07 00:41:10 +01:00
After [installing Go ](https://go.dev/doc/install ) and doing a `go mod download` to install all required prerequisites, just like the [Dockerfile ](Dockerfile ) does, you can:
```bash
make test # go test -v ./...
go build ./cmd/mc-router/
```
2023-09-09 19:06:59 +02:00
### Skaffold
2021-11-20 19:16:56 +01:00
For "in-cluster development" it's convenient to use https://skaffold.dev. Any changes to Go source code
will trigger a go build, new container image pushed to registry with a new tag, and refresh in Kubernetes
2022-01-07 00:41:10 +01:00
with the image tag used in the deployment transparently updated to the new tag and thus new pod created pulling new images,
as configured by [skaffold.yaml ](skaffold.yaml ):
2021-11-20 19:16:56 +01:00
skaffold dev
When using Google Cloud (GCP), first create a _Docker Artifact Registry_ ,
2022-01-07 00:41:10 +01:00
then add the _Artifact Registry Reader_ Role to the _Compute Engine default service account_ of your _GKE `clusterService` Account_ (to avoid error like "container mc-router is waiting to start: ...-docker.pkg.dev/... can't be pulled"),
2021-11-20 19:16:56 +01:00
then use e.g. `gcloud auth configure-docker europe-docker.pkg.dev` or equivalent one time (to create a `~/.docker/config.json` ),
and then use e.g. `--default-repo=europe-docker.pkg.dev/YOUR-PROJECT/YOUR-ARTIFACT-REGISTRY` option for `skaffold dev` .
2023-09-09 19:06:59 +02:00
### Performing snapshot release with Docker
2019-07-15 00:02:12 +02:00
```bash
docker run -it --rm \
-v ${PWD}:/build -w /build \
-v /var/run/docker.sock:/var/run/docker.sock \
goreleaser/goreleaser \
release --snapshot --rm-dist
2020-12-29 17:50:43 +01:00
```
2021-11-08 15:44:07 +01:00
2023-09-09 19:06:59 +02:00
## Related Projects
2021-11-08 15:44:07 +01:00
2024-08-05 00:22:39 +02:00
* https://github.com/haveachin/infrared