Skip to content

Latest commit

 

History

History
120 lines (90 loc) · 12.4 KB

readme.md

File metadata and controls

120 lines (90 loc) · 12.4 KB

Spring Boot based FAF-Server

Codacy Badge Build Status Coveralls Status

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)

How to run

Prerequisites

In order to run this software, you need to set up a FAF database.

From source

In order to run the application from source code:

  1. Clone the repository
  2. 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)
  3. Configure your JDK 8 if you haven't already
  4. Make sure you have the IntelliJ Lombok plugin installed
  5. Launch FafServerApplication

From binary

Given the number of required configuration values, it's easiest to run the server using faf-stack:

docker-compose up -d faf-server

Technology Stack

This project uses:

Architecture

Architecture overview

Learn

Learn about Spring Integration: https://www.youtube.com/watch?v=icIosLjHu3I&list=PLr2Nvl0YJxI5-QasO8XY5m8Fy34kG-YF2

Why I made this

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:

  • Reuse mature and stable technology instead of building "our own thing"
  • Reduce the amount of code to be written
  • Reduce possible of bugs and vulnerabilities
  • Be easily adjustable for future requirements

Doesn't use any framework, so that it:

  • Reinvents the wheel
  • Requires much more code to be written
  • Is more error-prone
  • Doesn't adjust well for future requirements

Uses a statically typed programming language which:

  • Drastically reduces the amount of possible bugs
  • Allows to find many bugs before the software is even started
  • Makes it a lot easier for developers to find their way around

Uses a dynamically typed programming language which:

  • Allows the developer to do things that make no sense at all
  • Makes it very difficult to impossible to automatically detect bugs, so you often only find them when the software is running (even very trivial ones like typos)
  • Makes it much harder for (especially new) developers to find their way around

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.

Solved problems of the current server

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.

Additional features

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