Replies: 9 comments
-
This is not something that is only achievable via a DB. I feel it very necessary that we don't conflate "using file I/O" with "using Minecraft's Anvil format". I am still not convinced of the fact that we need a DB. But I think discussion around this is good. I also think that supporting the original Anvil file format is a must, if we want to actually see adoption. But this should be togglable in config, to instead use Pumpkins format, whichever we choose to make it. Supporting saving files in the Anvil file format should not be development priority though, our own world-saving method should be a the gold standard in our code base. |
Beta Was this translation helpful? Give feedback.
-
My assumption that we will be using MCA region files came from @Snowiiii words on discord:
I agree that we could create our own binary format, but that would mean reinventing the wheel. That being said, I'm happy to hear any ideas you have for improving the Pumpkin format. I think we're still missing the bigger issue here, which is handling extreme parallel workloads with thousands of players. One key benefit of a single file mmap database is not just I/O efficiency, but also that the OS caches the file descriptor and its memory region, making memory mapping essentially 'free'. If we go with a file-based approach and try to use memory mapping, we'd still incur the cost of filesystem lookups and page faults, which will slow things down. |
Beta Was this translation helpful? Give feedback.
-
Sounds like a cool idea. I'm not against using a database approche as long User's can export their World files. Im fine if you want to start implementing db storage, Im interested to see real Performance benchmarks between databases and just normal OS I/O (e.g. Windows, Linux). |
Beta Was this translation helpful? Give feedback.
-
Is the Anvil ".mca" format compressed at all? If not, then our wheel could be significantly better. |
Beta Was this translation helpful? Give feedback.
-
Yes it is. https://minecraft.wiki/w/Region_file_format#Payload |
Beta Was this translation helpful? Give feedback.
-
It can be. It also can be not |
Beta Was this translation helpful? Give feedback.
-
@Asurar0 do you have any data to back up any of these claims? If not, this is just speculative premature optimization, and we should wait until we have data that confirms that the OS / file system is a bottleneck for large servers. I'm also confused by what some of your claims mean.
What do you mean by this? Are you suggesting that the OS slows down when you have more file descriptors open? Are you referring to the fact that the OS may have a limit for the number of file descriptors that can be open at once?
I don't understand what you mean by this. Blocking file I/O affects both the Anvil and DB approaches. And in fact, the problem is kind of worse when you use memory mapping, because then you have no control over when your threads block due to page faults.
This is true without memory mapping as well. Any recently accessed anvil files will be in the OS page cache.
You'll have to incur the cost of page faults with the single file DB just as much as you would with many files, unless the DB is substantially more space efficient. It all depends on how much data the OS can keep in the page cache. The file system lookups, though, sound like a plausible source of slowdown.
I've never heard of this as a solution for improving the performance of loading and unloading hundreds of files, but I'm curious to know more. Can you provide any sources for me to read on this? |
Beta Was this translation helpful? Give feedback.
-
I just stumbled upon this and I have multiple Questions. If you use a Database, why not go with an SQL compatible one, to be able to tap into a greater Ecosystem, allowing multiple Vendors, Configurations and/or even external Databases? Why B+ Trees, if you want to access an chunk, you know exactly where it is, hence you are able to use an hash index (for max cache util a locality-sensitive one), which has superior lookup times. Storing chunkdata in files, roughly equals a hash based database (-the Featureset). One of the strongest arguments for using an off-the-shelf RDBMS is the ecosystem, for a small server you can just use an SQL-Lite database, for larger servers you can use postgres and for the largest, you can even use a cluster and caching out of the box. Using an organised Format would allow Devs to write (Web-)Apps for e.g. Worldmaps, Scoreboards and so on, without even touching the Server-Code. If you want to use a Database, why not an SQL based one? some comments from an outsider, thank you all |
Beta Was this translation helpful? Give feedback.
-
I am not sure on the MCA format and the benefits of storing that data in a database, but some things like the ban lists, whitelists, etc. they are all in plain JSON format. For example, to add a new object, you have to bring the entire JSON doc into memory just to append a new entry. Even a simple database like SQLite only brings the pages the query has to deal with in memory. So adding a new entry for an IP ban would be super performant. You could also design the data model, indexes, and views, to be optimized for the known access patterns of checking if someone's IP or user is banned, adding a new entry, and so on. IMO, it'd not be a huge lift to make this optimization early and if need be migrate this to a different DB later. Storing game state is very different from the, but it is getting ahead of the problem. Before a database is chosen, be it SQL, NoSQL, or whatever, there has to be a target for what the server has to support. What is the MVP, and what does success look like? Even better would be to have targets for times too. For example, it should take no more than XXms from initial connection to entry into the world, and it should support 400 of those simultaneously. How many users should the server support concurrently? and so on. Otherwise, we risk solving for a problem that will not arise in the vast majority of uses and overcomplicate smaller installs. I saw it mentioned in one area about running on a tiny RaspPi, which is fantastic, but that is different solution than 50,000 players in a world and keeping all of game state going. There needs to be a product definition in this regard before technology solutions are chosen. |
Beta Was this translation helpful? Give feedback.
-
Introduction
Minecraft employs a dedicated file format, known as the Anvil file format, to store chunk regions on disk for subsequent retrieval. This file storage approach is consistently utilized across all Minecraft Java Edition server implementations.
This issue aims to discuss the advantages of utilizing a B+Tree database (notably LMDB) over a file I/O approach for storing the world, as well as provide tools for converting between these storage formats, specifically exporting .MCA files from a database and importing .MCA files into a database.
Assumptions
The following assumptions underlie the reasoning of this proposal:
mmap()
function in POSIX, whereas "I/O" refers to classical input/output operations over file descriptors (or equivalent), exemplified by theread()
function in POSIX.File I/O
The File I/O approach, utilized by the standard Minecraft Java Edition server implementation, involves storing data in a dedicated folder containing .MCA files, each named according to the location of the 32x32 chunk region it represents:
Pumpkin can derive the region coordinate from a specific chunk location as follows:
The logic for interacting with chunks involves the following steps:
Certain steps can be optimized. For instance, by calculating the distance between the player and the edge of the region, you can determine if it is less than the render distance. If so, you can proactively fetch both region files simultaneously, reducing the need for subsequent region file loads.
Pros:
Cons:
Database Approach
The database approach utilizes LMDB, a B+Tree-based database, to store chunk information in tables, enabling logarithmic time retrieval and updates. This is the approach used
by FerrumC project.
The logic for interacting with chunks involves the following steps:
The performance of this approach also depends on the implementation of the database, moreover, parallelization of reads and single-threaded writes over a dedicated threadpool.
Why LMDB ?
LMDB is considered a gold standard due to its exceptional performance and reliability, surpassing that of other options in the ecosystem. MDBX is an improvement over LMDB, but its Rust crate support is unfortunately declining.
Pros:
Cons:
Context
This proposal follows my previous experience assisting FerrumC in migrating from RocksDB (LSM structure) to LMDB (B+Tree structure), and aims to address and clarify any misconceptions surrounding the original proposal I made on Discord.
Conclusion
The MCA Region file approach is a proven and suitable solution for small to medium-sized servers, running on consumer or dedicated hardware, respectively. However, Pumpkin aims to achieve performance and efficiency that surpasses even the largest single-map servers, such as Folia and MultiPaper, which have successfully handled around 1000 simultaneous players. Pumpkin's goal is to exceed this benchmark, targeting player counts of over 1500 on a single dedicated hardware setup, a scale that would push the limits of file-based approaches. It is well understood that file-based approaches have inherent limitations when it comes to handling high parallel workloads.
This issue propose three options for Pumpkin:
Beta Was this translation helpful? Give feedback.
All reactions