Let's talk train design

Don't know how to use a machine? Looking for efficient setups? Stuck in a mission?
Tertius
Smart Inserter
Smart Inserter
Posts: 1450
Joined: Fri Mar 19, 2021 5:58 pm
Contact:

Re: Let's talk train design

Post by Tertius »

waterBear wrote: Thu Sep 25, 2025 2:30 am Yeah, that's the dream. I think you can probably do it with a centralized system, but I am lazy and don't want to build (another) huge combinator brain. Good luck though!

As for the trip planning, I do everything I can to let the game mechanics do that for me. One of my tricks is to attach a single combinator to each stop that basically says "set priority to 50 minus C" where C is the number of trains on the way or at the stop. That automagically prevents the trains from all going to the nearest pickup and ignoring distant providers, for example. It also works for delivery stations. Sometimes you have multiple ore drop offs that all need to be evenly fed, and this trick solves that problem nicely - with one combinator.

It's one of the reasons I don't like a lot of my own solution ideas so far. I want to avoid a system that tells the trains where to go via signal, both because that's going to end up being a mess of combinators and also because then I have to do that work instead of the game engine.
You're describing the system I presented. In detail. It has all the properties you envision. You even did the same priority calculation I do. I literally do the same: an arithmetic combinator with P=50 - C. Everything else is being done by the game engine with all the interrupts I described.

The only thing we differ is "multiple ore drop offs that all need to be evenly fed". The general solution to this is not a circuit but instead overproviding ore. Produce more ore than is being consumed, so full trains will accumulate in the long run, so there will always be a full train available for any drop off station, so they're being filled by a waterfall of trains automatically.
Since your factory will always grow, never shrink, there can never be any "too big" ore supply, since you need to build more ore loading stations anyway.
If you must underprovide, you can use a circuit to disable any unloading station until its buffer chests are below a threshold. Using priority even for unloading stations instead of disabling or just relying on train limits as I do is a neat idea by @Harb42, I guess I will combine this with the buffer chest fill state on my own maps. This will help during initial ramp up. Thanks for that idea!
waterBear wrote: Wed Sep 24, 2025 10:57 pm The basic idea for the most common implementation is this: A requesting stop globally broadcasts a needed number of train loads of an item. Provider stops then come online, and trains go straight to them.

There are lots of "gotchas" with the way trains and interrupts work that make this harder than it sounds in practice. It's do-able if you have one provider station for every item. By playing with the C signal from the stops, you can ensure that global demand gets reduced by 1 for every train on its way to either a provider stop for the item or to the requesting stop.
to help solve this problem, I'll take my work back offline.
By calculating this way, you get unavoidable edge cases. It's not possible to reduce the amount of the global demand and react on it the same tick. There is a minimum delay of 1 tick. In the 0th tick a train reserves a station, so its from C=0 to C=1. This means, global demand is lowered by 1, because the train was just dispatched. However, you cannot react in the same tick, because if there is a circuit reacting to the value of C, its reaction is visible only in the next tick, so for one tick there is a wrong value for the global demand. Depending on how important that value is for whatever, this can be a problem. And it will hit in the long run. Not perhaps during development, but later if you run it for days with a larger amount of trains.

This can happen for everything: If you use circuits, there is always a delay between a status change and the reaction to that status change. The delay is absolutely unavoidable, minimum 1 but usually 2 or more depending on intermediate combinators, however some built in game mechanics don't have this delay but react instantly such as reserving a station that just got empty. So some things just cannot be controlled and managed correctly all time with circuits.
Harb42
Inserter
Inserter
Posts: 27
Joined: Thu Oct 31, 2024 11:38 pm
Contact:

Re: Let's talk train design

Post by Harb42 »

computeraddict wrote: Thu Sep 25, 2025 6:45 am
Harb42 wrote: Thu Sep 25, 2025 3:52 am By some time all fluid trains will carry water which become basically useless.
That means you're doing it wrong, not that the mechanics have flaws. All of the solutions described in this thread cannot have this happen.
Yes, it is easily can happen. In addition, it is explicity stated in the FFF-s, that this is kind-of intended: it goes to whatever loader it is, not only what type. Read back.
Harb42
Inserter
Inserter
Posts: 27
Joined: Thu Oct 31, 2024 11:38 pm
Contact:

Re: Let's talk train design

Post by Harb42 »

Tertius wrote: Thu Sep 25, 2025 8:45 am
Harb42 wrote: Thu Sep 25, 2025 3:52 am My main problem is the following:
Your build contains a mistake, based on wrong assumption about the wildcard expansion. Use it for unloading, but not for loading.

The idea is to start with an empty train. Completely empty. For an empty train, no green cargo or fluid wildcard can match. So you cannot use the wildcard for the loading process. So instead of using an interrupt for loading, add the loading as the only static schedule entry. For this to work, you need to name all loading stations the same, no matter the material it's supposed to load. Instead of "(fluid type) load", name all of them just "fluid load". Overcome yourself and just equalize all fluid loading station names. And do it accordingly for the cargo item transport trains, name them all "cargo load".

So any empty train will go to the nearest free "fluid load" station and starts loading until it is full.
Now the interrupt condition (fluid wildcard) > 0 can trigger and will match the fluid just loaded. The waiting condition it creates should be (fluid wildcard) = 0 so it is ensured the train is completely empty after it leaves the unloading station, so the cycle will start again with an empty train on the static schedule entry. The train must be completely empty, not almost empty as you did.

Only if you completely empty a train this train system is universal. One cycle it is being loaded with water, the next cycle it might be filled with oil. This cannot happen if you keep some leftover. The train has to lose its identity to be available for everything the next cycle.

In case you see just the nearest fluid load station being visited and just the same fluid is available, you need to distribute more trains. If you have enough trains, the nearest station will always a train, either at the station or driving there, and the next train will reserve the next near station, and so on, until all loading stations are occupied.
The amount of trains with the same fluid will be implicitly limited by the sum of the train limits of the loading station of that fluid and the train limits of the unloading stations of that fluid. So if you have train limit 1 and 3 stations loading water and 1 water unloading station, you will see there will never be more than 4 trains loaded with water. So you add 4 trains for water.
So the 5th train cannot be loaded with water, it is available for loading with the next fluid. If you have a similar setup for the next fluid, you need to add 4 trains for the next fluid as well.
The trains themselves don't have an identity, they change fluids dynamically.
Hmm. You are right. It is not straightforward that it does not work for loading. -> I think it can be made that way, that it does not have to empty, so it "remember" what to load and unload all the time, because the cargo is not empty. "I got oil, okay I start to unload it. I am almost empty, time to load. What to load? Okay, I got oil, I go for loading oil." I think this 'attached' (and still independent) operation can be really useful to guarantee specific resource. For example the stone is really far away, and I always want at least 1 stone cargo, but don't want to create a new train group for it.
Besides this, you are right, in the beginning the train is empty, so it does not know what to load. This would be a manual step, and that's it.

The other think, the logical error in my setup: What you stated only works if the loader the train never goes back to park. Because in that case the "slot" will be freed up, and can consume trains indefinitely. That was my design's flaw. And for the unloaders it is advised to go back to the dock (if all the loaders are full), because it would create a deadlock otherwise. Loaders wait the Unloader stations to be empty, Unloaders wait the Loader station to be empty.

So, thanks for the insights, I go and try it out :)
jaxmed2
Burner Inserter
Burner Inserter
Posts: 6
Joined: Wed Sep 24, 2025 2:07 am
Contact:

Re: Let's talk train design

Post by jaxmed2 »

Harb42 wrote: Thu Sep 25, 2025 6:14 pm Hmm. You are right. It is not straightforward that it does not work for loading. -> I think it can be made that way, that it does not have to empty, so it "remember" what to load and unload all the time, because the cargo is not empty. "I got oil, okay I start to unload it. I am almost empty, time to load. What to load? Okay, I got oil, I go for loading oil." I think this 'attached' (and still independent) operation can be really useful to guarantee specific resource. For example the stone is really far away, and I always want at least 1 stone cargo, but don't want to create a new train group for it.
Besides this, you are right, in the beginning the train is empty, so it does not know what to load. This would be a manual step, and that's it.
I have to ask (maybe I'm still misunderstanding something?) but if having a train "remember" what it's loading is so important, why even fuss around with groups and interrupts? In fact the very point of the groups and interrupts is not having to do things like make a whole new group for every new resource. The pre-2.0 way of doing things using dedicated trains for each resource type might be more what you're after?

Just for completeness and at the risk of stating the obvious, there are two main ways of having "logistical" (not quite LTN but still allowing for a generic-ish requester/provider setup) way of doing trains:

Pre-2.0: Each resource type (e.g. oil) gets one or more dedicated trains. Each train has a very similar schedule: go to "oil pickup" until full, go to "oil dropoff" until empty, repeat. Add dedicated trains for each resource type with an appropriate schedule as you scale up. But this setup means you have to think more about how to keep your trains from running out of fuel.

Post-2.0: You can mix & match resource types but you should still have a separate train group for each train configuration (e.g. "liquid 1-1-1" vs "cargo 2-6" and so on.) Each train group has the following interrupts:
1. When cargo is empty, go to, e.g. "liquid 1-1-1 pickup" (whatever matches this train group) until full <-- important NOT to use wildcard here, provider station names are HARDCODED for each possible train group!
2. When cargo is full, go to "<wildcard> 1-1-1 dropoff" until empty <-- here is where the wildcard comes in, but also still good to use slightly different names for each train group in case you want to have multiple train sizes on the same network (e.g. support both "1-4-1 coal requesters" and "2-8 coal requesters" in the same network)
3. If fuel is low AND cargo is empty, go to the appropriate refuel station for this train size & shape until all locomotives are fueled <-- this interrupt can interrupt other interrupts
4. If destination is full/no destination AND cargo is empty, go to a depot <-- this interrupt can interrupt other interrupts
Add generic trains to each corresponding train group as you scale up.

Both setups of course assume that each station has a circuit condition to enable/disable itself (or set a dynamic train limit if you want a train buffer at a given station) based on their supply buffers.

So long as you have enough trains (which really is the key), either solution basically gives you a requester/provider setup. The main caveat is that you basically must add a new train for each additional provider station "slot" (including buffers) that you build, otherwise you do run the risk of having all of your trains congregate at one resource while other resources get starved. But again that's easily solved by simply having enough trains.

I can only see needing anything more complex than this if there are some niche optimizations you want to make: like the "50-C" dynamic priority if all your stations support dynamic limits and you want to load balance your train buffers. But otherwise you'll get the expected behavior of "trains go to providers, fill up, wait until a requester station opens up, make the delivery, repeat while keeping myself fueled up and never block requester stations" pretty easily using built-in mechanics. Any further optimizations like "I don't want my trains to even go the provider station unless that resource actually needs a delivery" to me just sound like a symptom of "I don't have enough trains"
waterBear
Fast Inserter
Fast Inserter
Posts: 123
Joined: Thu Apr 23, 2020 2:05 pm
Contact:

Re: Let's talk train design

Post by waterBear »

jaxmed2 wrote: Thu Sep 25, 2025 9:42 pm
Harb42 wrote: Thu Sep 25, 2025 6:14 pm Hmm. You are right. It is not straightforward that it does not work for loading. -> I think it can be made that way, that it does not have to empty, so it "remember" what to load and unload all the time, because the cargo is not empty. "I got oil, okay I start to unload it. I am almost empty, time to load. What to load? Okay, I got oil, I go for loading oil." I think this 'attached' (and still independent) operation can be really useful to guarantee specific resource. For example the stone is really far away, and I always want at least 1 stone cargo, but don't want to create a new train group for it.
Besides this, you are right, in the beginning the train is empty, so it does not know what to load. This would be a manual step, and that's it.
I have to ask (maybe I'm still misunderstanding something?) but if having a train "remember" what it's loading is so important, why even fuss around with groups and interrupts? In fact the very point of the groups and interrupts is not having to do things like make a whole new group for every new resource. The pre-2.0 way of doing things using dedicated trains for each resource type might be more what you're after?

Just for completeness and at the risk of stating the obvious, there are two main ways of having "logistical" (not quite LTN but still allowing for a generic-ish requester/provider setup) way of doing trains:

Pre-2.0: Each resource type (e.g. oil) gets one or more dedicated trains. Each train has a very similar schedule: go to "oil pickup" until full, go to "oil dropoff" until empty, repeat. Add dedicated trains for each resource type with an appropriate schedule as you scale up. But this setup means you have to think more about how to keep your trains from running out of fuel.

Post-2.0: You can mix & match resource types but you should still have a separate train group for each train configuration (e.g. "liquid 1-1-1" vs "cargo 2-6" and so on.) Each train group has the following interrupts:
1. When cargo is empty, go to, e.g. "liquid 1-1-1 pickup" (whatever matches this train group) until full <-- important NOT to use wildcard here, provider station names are HARDCODED for each possible train group!
2. When cargo is full, go to "<wildcard> 1-1-1 dropoff" until empty <-- here is where the wildcard comes in, but also still good to use slightly different names for each train group in case you want to have multiple train sizes on the same network (e.g. support both "1-4-1 coal requesters" and "2-8 coal requesters" in the same network)
3. If fuel is low AND cargo is empty, go to the appropriate refuel station for this train size & shape until all locomotives are fueled <-- this interrupt can interrupt other interrupts
4. If destination is full/no destination AND cargo is empty, go to a depot <-- this interrupt can interrupt other interrupts
Add generic trains to each corresponding train group as you scale up.

Both setups of course assume that each station has a circuit condition to enable/disable itself (or set a dynamic train limit if you want a train buffer at a given station) based on their supply buffers.

So long as you have enough trains (which really is the key), either solution basically gives you a requester/provider setup. The main caveat is that you basically must add a new train for each additional provider station "slot" (including buffers) that you build, otherwise you do run the risk of having all of your trains congregate at one resource while other resources get starved. But again that's easily solved by simply having enough trains.

I can only see needing anything more complex than this if there are some niche optimizations you want to make: like the "50-C" dynamic priority if all your stations support dynamic limits and you want to load balance your train buffers. But otherwise you'll get the expected behavior of "trains go to providers, fill up, wait until a requester station opens up, make the delivery, repeat while keeping myself fueled up and never block requester stations" pretty easily using built-in mechanics. Any further optimizations like "I don't want my trains to even go the provider station unless that resource actually needs a delivery" to me just sound like a symptom of "I don't have enough trains"
This is basically what I do on Nauvis. Habit from before Space Age, adapted to a few of the new tools. I have a group for each resource (there are really only a handful of resources you have to make a group for). Schedule is: go to pickup until full, go to drop off until empty, and a refuel interrupt.

There are exceptions, like the wall resupply trains which have some logic to decide when to show up and what to unload, but those are all unsurprising I think.

There are other special case trains, but the time I spend setting up trains this way is such an infinitesimally tiny percentage of the time I spend building stuff that I never bothered to change my ways.
Tertius
Smart Inserter
Smart Inserter
Posts: 1450
Joined: Fri Mar 19, 2021 5:58 pm
Contact:

Re: Let's talk train design

Post by Tertius »

waterBear wrote: Wed Sep 24, 2025 9:20 pm Have you ever wished that your trains could behave like logistic robots? If so, then that's what this idea is.
I would like to answer this as well. No, I never wished trains could behave like logistic robots.

Logistic robots are inefficient if you just place a roboport for area coverage and let the bots do their thing. It's inefficient, because for continuous higher volume transports the robots pile up at the destination where nobody need them. In case a transport order is raised, the robot has to fly to the chest for pickup, picks up, then goes to recharge since its battery became half empty by flying here, wait in the recharging queue, and only then it will actually go to deliver the item.

To make this kind of regular transport more efficient, you need to tweak your logistics network. The first thing you need to do is to place buffer chests near the place where you need item availability, so items are available immediately for a local short distance transport. For short distance transport up to 10-15 tiles, there is nothing faster than delivery by bot, not even quality stack inserters.

The second thing you need to do is place roboports near pickup places. Directly next to the buffer chests and near the places where items are being crafted and made available. And you need to configure the roboports to call some amount of logistic robots into that roboport. This way robots return to the pickup places in advance before a pickup order is actually being raised, so they're available for immediate pickup and they have full battery charge. There is a 3rd thing to do, however it's not relevant for the train comparison: in case of higher transport volume, you need to place multiple roboports next to each other to provide enough local charging spots, so the robots are able to charge on a local roboport and don't need to fly away or queue up for recharge. All this is relevant and very visible at least on Aquilo if you want to feed multiple rockets with the Aquilo science packs simultaneously and as fast as possible by bots.

And this item and robot buffering to decrease the latency between raising the order and actually getting the delivery by train is what you do by using the vanilla implementation I described. In this implementation, the time a station is idle is so short (just 5-10 seconds if you use train limit 2 with a waiting area), you don't even need buffer chests, so the items in flight is just 1-2 train loads - and that's just 21-42% of a full station with 4 full buffer chests per wagon.
Premu
Fast Inserter
Fast Inserter
Posts: 159
Joined: Wed Nov 13, 2019 4:40 pm
Contact:

Re: Let's talk train design

Post by Premu »

I just stumbled about this topic. And I actually set up a trainsystem with that idea:

- Empty trains wait central depot stations.
- Requester stations will only request as many trains as they can actually empty out.
- Provider stations will only be activated, if they have enough to supply a train AND if there is one potential requester station ready to take on the train load.

It involved a lot of circuitry to realize such a system, and a lot of fine tuning to catch potential borderline cases. And there are many of those.

My solution was to have a central coordinator circuit for each good in my train network which receives requests from requester stations and enables provider stations. It will only enable one provider at one time, so the system can handle multiple providers as well as multiple requesters of the same good. It's took me a long time to fix all problems, but finally it seems that there are no issues left. It's not a very robust system, though - if you mess up something while setting up a new station, strange things will happen, and it can be a mess to repair. And it's also not really the most efficient for pure throughput. I mostly did that as challenge for myself.

But it was possible just with vanilla elements, so with enough effort you can do almost everything.
waterBear
Fast Inserter
Fast Inserter
Posts: 123
Joined: Thu Apr 23, 2020 2:05 pm
Contact:

Re: Let's talk train design

Post by waterBear »

Premu wrote: Fri Sep 26, 2025 8:56 pm I just stumbled about this topic. And I actually set up a trainsystem with that idea:

- Empty trains wait central depot stations.
- Requester stations will only request as many trains as they can actually empty out.
- Provider stations will only be activated, if they have enough to supply a train AND if there is one potential requester station ready to take on the train load.

It involved a lot of circuitry to realize such a system, and a lot of fine tuning to catch potential borderline cases. And there are many of those.

My solution was to have a central coordinator circuit for each good in my train network which receives requests from requester stations and enables provider stations. It will only enable one provider at one time, so the system can handle multiple providers as well as multiple requesters of the same good. It's took me a long time to fix all problems, but finally it seems that there are no issues left. It's not a very robust system, though - if you mess up something while setting up a new station, strange things will happen, and it can be a mess to repair. And it's also not really the most efficient for pure throughput. I mostly did that as challenge for myself.

But it was possible just with vanilla elements, so with enough effort you can do almost everything.
Yeah, that's more or less the route I've been investigating. A central system. The thing you really need is to make the dispatch mechanism sequential, which is kind of a shame because throughput gets lost like you said. The system I was experimenting with was dispatching once every 3 frames which seemed like plenty of throughput for my needs, but by the time all the kinks are out I bet that rate drops a lot.

Like you said, there are a lot of bugs. I'm glad to hear you got it to work. I'm not sure I have the willpower to keep working on it. It's been several days and I just want Fulgora working again 😂
Harb42
Inserter
Inserter
Posts: 27
Joined: Thu Oct 31, 2024 11:38 pm
Contact:

Re: Let's talk train design

Post by Harb42 »

jaxmed2 wrote: Thu Sep 25, 2025 9:42 pm
Harb42 wrote: Thu Sep 25, 2025 6:14 pm Hmm. You are right. It is not straightforward that it does not work for loading. -> I think it can be made that way, that it does not have to empty, so it "remember" what to load and unload all the time, because the cargo is not empty. "I got oil, okay I start to unload it. I am almost empty, time to load. What to load? Okay, I got oil, I go for loading oil." I think this 'attached' (and still independent) operation can be really useful to guarantee specific resource. For example the stone is really far away, and I always want at least 1 stone cargo, but don't want to create a new train group for it.
Besides this, you are right, in the beginning the train is empty, so it does not know what to load. This would be a manual step, and that's it.
I have to ask (maybe I'm still misunderstanding something?) but if having a train "remember" what it's loading is so important, why even fuss around with groups and interrupts? In fact the very point of the groups and interrupts is not having to do things like make a whole new group for every new resource. The pre-2.0 way of doing things using dedicated trains for each resource type might be more what you're after?

<<< I deleted here the rest of the message to not waste space >>>
Thanks for asking the details!

Let me share you my history in a nutshell:
- I started to play the game from 1.1, a bit more than a year ago.
Never played this kind of builder/logistical games before. But others, so at least I have quite a gaming experience.
- I started out with my friends who installed bunch of mods which made the game quite easy: warehouse, more powerful belt tiers, max everything at planet start, peaceful mode, etc.
I though these make all the things too easy, killing the game itself.
- So I started out some new games using only QoL mods or some others which make the game more funky. E.g.: Helicopter, asphalt, etc.
I had 2 pre SA games and 1 SA game, this one. I did not reach megabase size bases, just 'enoughly large' bases.
- Recently I decided, okay, go megabase. Go hard or go home. I was aware of the main bus approach and grid based approach - whatever that means.
And I mostly played in single, not reading much about base designing on the internet, not using other people's blueprints (except balancers).
So I decided to use grid approach and use a kind-of fully 'microservice' style. Every block has standard size, standardized slots of inputs and outputs, and any block does only 1 thing. Green chips, concrete, whatever. Only 1 step. There are really few exceptions on this.
So after I created the templates it is really easy to add new blocks. Train rails and poles are standardized too, so just a few blueprints and clicks to create a new block.
To increase SPM I had to add more blocks. And so the number of trains grew. It is around 300 now. All of them was 1-2-1.
<And from here now we are getting to the main question>
All trains was dedicated to 1 resource. So I have around 40-50 groups. And adding more and more trains is tedious. And sometime the balance of transportation goes sideway. Suddenly 10-20 of something is missing, or is in abundance. So I introduced: if a requester station is at least semi full, disable it. If a requester station is starving (no input), raise alarm - so I now what is missing, because it is impossible to follow anymore what do I have and what not. Also, set trains to leave station after 60 seconds - I don't want to have trains everywhere with half full cargo waiting in stations. Because if they all leave at once, they would clog the train system - probably - however it is quite robust as I see.

So, increasing the size of the factory became really-really easy, however oversee the resource bottlenecks and trains not. Tedious and boring.
<And now we arrive to the current problem>
Let's don't have intermediate blocks, just do the sciences from raw, and that's it. Less blocks, less trains. I have the basic needs block already, so everything will be still enough and available. And go to fully interrupt based approach.
So, - as you stated earlier - let's say I have 20 copper stations nearby. In worst case I have to add 21 trains to surely get other resources. -That is not optimal, I have to figure out something else- So, I thought use full interrupt: this way I don't need 40-50 groups, and could have easily do a 'resource attach' to a train IF it does not depleted completely let's say iron. (I thought. I thought wildcard works both way, for fill and drain.) This way I have for example 20 trains, 3 for iron, 3 stone, 3 coal, 3 copper, and the rest 8 do whatever station it finds. Also, less train, less congestion.
So having 2 groups (solid and fluid) is much easier to mantain than like 50 groups (which is not really a problem anymore, because the new interrupted trains are 2-6-0 sized).

My goal here is if I start a new game do 'everything' from my blueprints in a most convenient way. So interrupted approach is really easy to use, however not so optimal in itself. So I tried to balance these things.

So, there is a learning curve here along the way and a migration as well from consecutive 'architectures'.
Post Reply

Return to “Gameplay Help”