This is a reimplementation of the Forged Alliance Forever's server application.
It aims to abstract the communication protocol as far as possible in order to stay compatible with current server's legacy protocol while at the same time allowing new protocols to be added and supported simultaneously.
As the underlying (legacy) database schema isn't based on best education and diligence either, some data types, structures, concepts and/or field names may be questionable, too, even though efforts are made to abstract it with a clean layer as well (work in progress)
In order to run this software, you need to set up a FAF database.
In order to run the application from source code:
- Clone the repository
- Import the project into IntelliJ. For some reason, IntelliJ deletes launch configurations after import. Please revert such deleted files first (Version Control (Alt+F9) -> Local Changes)
- Configure your JDK 8 if you haven't already
- Make sure you have the IntelliJ Lombok plugin installed
- Launch
FafServerApplication
Given the number of required configuration values, it's easiest to run the server using faf-stack
:
docker-compose up -d faf-server
This project uses:
- Java 8 as the programming language
- Spring Boot as a base framework
- Spring Integration as a messaging framework
- Gradle as a build automation tool
- Docker to deploy and run the application
Learn about Spring Integration: https://www.youtube.com/watch?v=icIosLjHu3I&list=PLr2Nvl0YJxI5-QasO8XY5m8Fy34kG-YF2
I have high expectations on software quality. There are many reasons why the current Python server, even though it has been refactored over months, still is of bad quality. So let's compare this Java based server to the current Python based server.
Java Server | Python Server |
---|---|
Has thoroughly been performance-tested, proofing that it can withstand more load than FAF will ever experience. | Has never been performance-tested during its development, later performance tests revealed that it handles load very badly and even crashes. |
Uses a very popular framework in order to:
|
Doesn't use any framework, so that it:
|
Uses a statically typed programming language which:
|
Uses a dynamically typed programming language which:
|
Makes use of a very strong and mature ecosystem when it comes to libraries, documentation, build tools, developer tools and everything you need. | Lives in an ecosystem that still has to be come mature, often lacks good documentation and has laughable build tools and dependency resolution. |
Is highly decoupled from the underlying communication protocol, so that additional protocols can be added easily and supported simultaneously. | Is strongly coupled to the underlying communication protocol so that it's difficult to add new protocols or support multiple protocols at once. Also the server's internal implementation can't be changed easily without breaking client compatibility. |
Makes heavy use of design patterns of modern Software development, making it easily understandable (for educated developers) and provides high flexibility while avoiding common mistakes. | Makes little use of design patterns, making it more difficult to understand, less flexible and more error-prone. |
Runs on Windows, Linux and Mac with very little setup time so that every developer can get started within couple of minutes. | Only runs on Linux based systems. In order to run it on Windows, developers have to spend a significant amount of time setting up a Linux virtual machine and installing many dependencies manually. By using Docker some problems are solved but replaced by new ones. |
Was built from scratch so it's free from any legacy bugs that are unknown or difficult to figure out. | Is a refactoring of a terrible code base so that there are many "left overs", and has given us ridiculous bugs nobody has yet figured out why they happen (see list of problems below). |
Can be managed while it is running (e.g. configuration values can be changed, or commands can be executed) using a web interface. | Can not be managed once it's running, instead it needs to be restarted for every configuration change. |
Produces very clear, well readable and helpful log messages making it easy for administrators to analyse and locate problems. | Produces a lot of unreadable, messy and useless log messages, making it difficult for administrators to analyse and locate problems. |
Was built in accordance with a set of design and implementation principles, like Clean Code, and strong quality control to assure high quality and maintainability. | Was built by "Hackers" with little principles or bigger picture in mind, focusing on "getting the job done". |
Is actively maintained by me, and various other people are (interested in) contributing. | Hasn't had a committed maintainer for over a year, and nobody is willing take over. Many unfinished PRs are lying around as the original authors lost interest, and nobody is taking care. |
Uses a database abstraction technology so that the application is decoupled from the underlying database, which allows easy switching to a different database. | Is tightly coupled to the rather infamous MySQL database, making it expensive to ever switch to different (better) database. |
Verifies incoming messages thoroughly. If a messages is invalid in the current context (e. g. a player reports a result for a game he isn't part of, or for an army that doesn't exist) he receives a very clear error message. This protects the server from malicious messages and helps client developers know when their application is misbehaving. | Does little message verification, allowing players to manipulate the system. Silently drops invalid messages or disconnects the client without specifying a reason, making it difficult for client developers to identify mistakes. |
The following issues that exist in the original server are not present in this implementation. Even though some of them have only "recently" been reported, they have all been around for at least 1.5 years.
- FAForever/server#116 Not all online players are reported
- FAForever/server#142 Server misreporting gameInfo to clients
- FAForever/server#166 Server bothers with whitespace in json messages
- FAForever/server#193 Explicit dependency versions
- FAForever/server#195 Coop Leaderboards [not updating]
- FAForever/server#200 Ingame swords still display in aeolus
- FAForever/server#213 Games without game result will still count as rated
- FAForever/server#224 Player reported as "in game" even though they're not
- FAForever/server#225 Players in lobby are not always detected properly
- FAForever/server#253 Player stats sometimes not stored
- FAForever/server#263 Game duration isn't stored correctly (too long)
- FAForever/server#265 Rating is always calculated based on player rating when game started
- FAForever/server#276 5-15 Zombie-games per day since server (re)start
- FAForever/server#282 Game titles seem to be quoted needlessly
- FAForever/server#283 Wrong game result
- FAForever/server#286 Auto-remove inactives from leaderboards
- FAForever/server#287 'Host has left the game' when trying to join a game
- FAForever/server#288 Explicitly set startTime
- FAForever/server#302 Unique id handling code does not handle malformed UIDs very well
- FAForever/server#316 Game stats and player ratings aren't updated before last player disconnects
- FAForever/server#319 Make ladder map selection non-random
- FAForever/server#323 Notice from server prompt spacing
- FAForever/server#343 If teams are unlocked the game should not be valid
This implementation provides the following additional features over the original server:
- Management and monitoring using a web interface
- Automatic update of the GeoIP file (used to display the country flags in the client)
- Support for min/max rating for games
- Verification that it's compatible with the underlying database schema version
- Updating scores for the league & divisions system after each ladder game
- ICE support
- Every error message has an error code that allows specific handling
- Allows connection via WebSocket
- Supports OAuth so service applications can connect as well
- Detailed logging which can be configured at runtime
- Allows disabling authentication, making it much easier for users to test