• We have updated our Community Code of Conduct. Please read through the new rules for the forum that are an integral part of Paradox Interactive’s User Agreement.

Stellaris Dev Diary #372 - Modding: Pop Groups and Jobs

Hi everyone!

This week, @Gruntsatwork will discuss the technical details of pop group scripting. This topic is likely to be of most interest to the modding community.

The systems we’re implementing now are just scratching the surface of where we want to go with them - we’re looking forward to some of the things we’ll be able to do with these tools over the next few years as well as seeing what you do with them.

As with all of these dev diaries, some of this is still subject to change during implementation and during the beta.

Pop Groups and Jobs​

Hello everyone, @Gruntsatwork here! Let us talk about some of the script changes that are coming with 4.0 when it comes to pops and jobs…

As Eladrin already mentioned in Dev Diary 370, we are changing the way we look at Pops by grouping them together into Pop Groups. These groups are defined by their species, traits, ethics, and factions but NOT their jobs. It is entirely possible and likely for a pop group to have pops working different jobs.

The goal is that for most purposes in the game, you will reference pop groups instead of pops, which should hopefully save us from iterating through every single pop in our empire whenever a modifier needs to be re-calculated.

This allows us to tone down or remove some of the most performance-intensive actions we used before and replace them with far more performance-friendly variations instead.

For example, that means that both “random_owned_pop” or “any_owned_pop” have been relegated to the dark corners of history and replaced with “random_owned_pop_group” and “any_owned_pop_group”.

The same holds true for many of the effects used on those pops, like create_pop or kill_pop, or move_pop. Going forward, we will now create, move and kill pop groups, either in their entirety or through percentages. And for the eagle-eyed among you, YES, that means you no longer have to loop through singular pops to do unspeakable things to them, you can nicely target their pop group and let it do the math for you.

Thanks to the tireless efforts of our programmers, who have given us some new functionality for scripted triggers like comparators, other old tools, like num_pops, will see a resurgence as a scripted trigger. We expect the modding folks to find a lot of use in that, even as we slightly dread what you will come up with.

As also mentioned by Eladrin, this means we no longer have a constant contact between pops and their job.

Instead, there is a single moment of assignment when the pop group briefly knows which job it is supplying and with how much workforce. From then on, the job only knows that it has been supplied with workforce and thus must produce the associated resource. As long as the assignment stands, we have no need to check on the pop again.

This brings us to one of our biggest changes: removing all production modifiers on species traits and replacing them with bonus workforce. Simply put, because the workforce assigned to a job does not know which species it came from during most checks, production bonuses from species (modifiers like +10% Research from Jobs from the Psionic trait) cannot be applied. Instead, species traits now provide modifiers like “+10% Bonus Workforce for Researcher Jobs”, which means 100 Psionic Pops working 100 Physicist jobs will have the job upkeep and output of 110 Physicist jobs. In other words, we only pay upkeep on 100 Pops, but we get the output of 110 Researchers! This also has the side effect of the modifiers for job output from species traits are now multiplicative with other modifiers.

As an example, in 3.14 if we had 1 Psionic (+10% Research from Jobs) Pop working a Researcher job in an empire with the Meritocracy civic (+10% Specialist Job Output) on a Relic world with a Central Spire (+15% Research from Jobs), the total output would be 3 × (1 + 0.1 + 0.1 + 0.15) = 3 × 1.35 = 4.05 of each type of research.

In 4.0 if we have 100 Psionic (+10% Bonus Workforce for Researcher Jobs) Pops working 100 Physics jobs in an empire with the Meritocracy civic (+10% Specialist Job Output) on a Relic world with a Central Spire (+15% Research from Jobs), the total output would be 3 × (1 + 0.1) × (1 + 0.1 + 0.15) = 3 × 1.1 × 1.25 = 4.125 physics research.

This brings us beautiful new script entries like this one from the Psionic trait:
A sample of a job that he really made as bad as possible.

It doesn’t have to be like this. But it can be. Grunts made his choice. - E
This is perfectly serviceable - G
I have hopes to refactor this - AS

As a side note, some checks can still query a pop’s job, but only indirectly, by scoping to a job and determining which pop group is filling it. This means we can still ensure functionality for Death Cults and similar targeted kill_pop effects.

In contrast, production bonuses on the planet or the empire are still available since they simply affect everyone.

So for species traits, we encourage the use of these new modifiers

pop_job_bonus_workforce_mult
To increase the bonus workforce a pop generates for a given job.

pop_job_workforce_mult
To increase the workforce a pop generates for a given job, this is not bonus workforce.

job_max_workforce_mult
To increase the maximum workforce a Job can accept

As a reminder, a job's workforce will fill to its maximum allowed but not beyond that. If a pop generates more workforce than usual, fewer pops will be required to fill the job to max, but it will not produce more than its maximum. If a pop generates a bonus workforce, it can go beyond the job's maximum and scale its production up.

In addition, we have also split quite a few of our economic categories that depended on triggered checks of species traits. This also includes the use of triggers to fake an inheritance of economic classes, which we have removed in many cases and only left in the ones we deemed the most reliant on them.

For inheritance, we recommend the normal parent-child structure of economic categories OR, to use static_modifiers to grant the modifiers of any combination of economic_categories.

Most, if not all, of these changes were made to improve performance: reducing calls, loops, and modifier cascades that would otherwise trigger recalculations across every planet and pop in your empire, just in case a deficit check was needed at that moment.

Looking ahead, we see great potential in workforce mechanics, both for us and the modding community. We've hinted at automation – workforce decoupled from pops – and some of you may have already considered new applications for Virtuality. Who knows what other, more extreme variations in the type and number of pops empires require might now be possible?

Simply put, we now have the workforce to power Stellaris for years to come. Pun very much intended.

What’s Next?​

Our planned livestream is going to be delayed a bit, and will likely end up being alongside the Open Beta. Right now our primary focus is on implementation.

Next week we’ll have some more updates on how things have been going.
 
  • 65Like
  • 21
  • 10Love
  • 3
  • 2Haha
  • 2
Reactions:
The same holds true for many of the effects used on those pops, like create_pop or kill_pop, or move_pop. Going forward, we will now create, move and kill pop groups, either in their entirety or through percentages.
Are you setting the default percentage to 0%?

I don't think we want people that forget the percentage to accidentally affect 100% once released. Better to have the frustration when writing and affecting 0%.

The goal is that for most purposes in the game, you will reference pop groups instead of pops, which should hopefully save us from iterating through every single pop in our empire whenever a modifier needs to be re-calculated.
Can one pop be in multiple pop groups, based on Point of View?
Like one Ethics Group, one Workforce Group?

Or will the Alloy Workforce group be split into dozens of subgroups based on Ethics, Stratum and what not?

As a reminder, a job's workforce will fill to its maximum allowed but not beyond that. If a pop generates more workforce than usual, fewer pops will be required to fill the job to max, but it will not produce more than its maximum. If a pop generates a bonus workforce, it can go beyond the job's maximum and scale its production up.
That last sentence is really confusing.
First you say Bonus workforce won't assign more then 100% workforce. Then you say it does?
Did you use the wrong words here?
 
This brings us to one of our biggest changes: removing all production modifiers on species traits and replacing them with bonus workforce. Simply put, because the workforce assigned to a job does not know which species it came from during most checks, production bonuses from species (modifiers like +10% Research from Jobs from the Psionic trait) cannot be applied. Instead, species traits now provide modifiers like “+10% Bonus Workforce for Researcher Jobs”, which means 100 Psionic Pops working 100 Physicist jobs will have the job upkeep and output of 110 Physicist jobs. In other words, we only pay upkeep on 100 Pops, but we get the output of 110 Researchers! This also has the side effect of the modifiers for job output from species traits are now multiplicative with other modifiers.
I hope the trait tooltips still just say "+10% food from jobs" because “+10% Bonus Workforce for Farmer Jobs” is a bit more wordy and honestly just kind of lame as a tooltip
 
  • 2
Reactions:
I hope the trait tooltips still just say "+10% food from jobs" because “+10% Bonus Workforce for Farmer Jobs” is a bit more wordy and honestly just kind of lame as a tooltip

But that would be inaccurate. They're not just giving you extra food; they're giving you extra throughput to a particular job category. You're leaving out the increased upkeep. Plus, it would incorrectly imply that non-Farmer jobs that produce food would be affected, as well as non-food output from Farmer jobs.

(Obviously, this doesn't make much sense for food and farmers, but it might come into play with more advanced resources).
 
  • 3
Reactions:
We mostly plan on using the non-bonus workforce modifiers for negative stuff, but for my example I'll also include a positive modifier. Imagine if there was a pair of traits Lazy (-50% Workforce) and Overachiever (+100% Workforce), and we have three species, Lazy Humans, Blorg, and Overachiever Starfish.

To fill 100 Miner jobs with each of the three species you would need:
  • 100 Blorg, they have no workforce modifiers so each Blorg contributes 1 Workforce.
  • 200 Lazy Humans, as each Lazy Human pop contributes 0.5 Workforce.
  • 50 Overachiever Starfish, as each Overachiever Starfish contributes 2 Workforce.

How would this workout if the lazy humans have 150% bonus workforce and the starfish have -50% bonus workforce?
(and for comparison 100%/200% humans and -25% Starfish)


Who will work the job in the end?
What will be the output?
 
Perhaps instead of saying "bonus workforce," in-game it could say "bonus work from jobs" or "bonus throughput from workforce" or something like that, to make it clear that it's not actually affecting the amount of pops working the jobs.
 
Last edited:
  • 1Like
Reactions:
Me and the boys getting ready for MP
View attachment 1259399
"Ahhh Mister Garibaldi"
I miss Londo Mollari and G'kar so much! Excellent actors, excellent acting, perfect script!
Seeing cast photos of Babylon 5 is always bit depressing considering how many of them are already dead.
The shadows took them as they will soon take us as well... I don't think we'll make it for any form of ascension here on earth - if anythign we're going backwards to a new dark age...
Any plans on giving Megastructures like the Dyson Sphere habitable areas? It feels really silly for such a structure to Just Work for eternity without regular maintenance, and for the people (or robots or whatever) not needing food/repair/amenities themselves.

It'd also be really cool for Dyson Sphere workers to suddenly realize they supply their Empire with 3/4 of their energy and get ideas...
Beyond the devs/Eladrins negative reply on this, I could assume the gigastructual mod team would try something like this. If it doesn't happen immediately, I plan to create a mod of habitable gigastructures, because I want to play with such things. Beware though, that I'm very busy and it could take months. This was a dream of mine for too long, but it couldn't be made properly and elegantly due to performance issues of the old/current pop system. To see what I mean of, try playing with a Birch World with Gigas.
Currently - not really. Right now we're still keeping it mostly single threaded and haven't gotten to many of the other possible optimizations yet, because it's a lot easier to understand what it's up to and debug this way.

Like I mentioned in another thread, we're likely to keep it this way during the early open beta so it's easier to fix any issues you guys hit, but I'm pretty confident about the new architecture being fundamentally better once we actually let it loose.
Back in the Mesozoic era when we where discussing performance here in that super merged and condensed megathread, me and others proposed a system where sectors and planets would be grouped and would be processed in paralled in seperate threads, and then have the results merged/synced together - this would result in scaling empire size and performance on raw CPU core count. It would be amazing! Is this being considered at all?

Sorry for late posting, had a hell of a day... I wish to have been here earlier!
 
  • 4Like
Reactions:
That last sentence is really confusing.
First you say Bonus workforce won't assign more then 100% workforce. Then you say it does?
Did you use the wrong words here?
My read is there are two variables on workforce. One varies how much workforce a pop supplies, the other applies a bonus on top of the supplied workforce.

So one trait might make a pop supply double the workforce in a job. In which case, you only need 50 pops to fill a 100 workforce job. So you pay upkeep for 50 pops and 100 workfirce, getting 100 workforce output.

Another trait provides 10% bonus workforce. So in the same job, you need 100 pops but get 110 workforce output. That's upkeep for 100 pops and 110 workforce, with 110 output.
 
Will beta be open to Mac players?
 
In contrast, production bonuses on the planet or the empire are still available since they simply affect everyone.

So for species traits, we encourage the use of these new modifiers

pop_job_bonus_workforce_mult
To increase the bonus workforce a pop generates for a given job.

pop_job_workforce_mult
To increase the workforce a pop generates for a given job, this is not bonus workforce.

job_max_workforce_mult
To increase the maximum workforce a Job can accept

As a reminder, a job's workforce will fill to its maximum allowed but not beyond that. If a pop generates more workforce than usual, fewer pops will be required to fill the job to max, but it will not produce more than its maximum. If a pop generates a bonus workforce, it can go beyond the job's maximum and scale its production up.
This is word salad and it needs to be cleaned up. Having a pop "workforce multiplier" and a "bonus workforce multiplier" is too confusing, unintuitive, and hard to remember. They need separate names.

One of these seems to be talking about per worker productivity. Value added per unit worker. That's the "bonus workforce multiplier". (also could be seen as work quality or skill)

The other seems to be talking about staffing. How many of these pop group workers are needed to complete the task. That's the "workforce multiplier", but it's more like a staffing ratio / headcount/ positions or something. (also could be seen as work rate)

I don't know which one you want to tweak the verbage on, maybe both. You guys have more context on where the words get used in other UI flows.

Right now, it just feels bad and like I'm going to fumble a species modification project to clunky tooltips.
 
  • 2Like
Reactions:
Bonus Thought:

Is it possible to have something like a job which consumes no staffing/use of pop groups; or more accurately, double dips on them?

Example: Corporate branch offices generate "targeted consumer" or "mark" jobs.
Every "mark" job generates +2 energy for branch office holder, but -20% happiness to the pop group and anti-materialist ethics shift.

--
Allowing this, or mods to do this, would let us model certain extractive, or beneficial, activities without dumping a fat dookie on the tuning of the job position/workforce counts and associated game systems.

It could also be reused for things like death cults with pop_kill targeting pop groups, or that player suggested idea of training jobs that "promote" their pops based on inverting the demotion system.

--
I guess I'm asking for a script flag which tells the game engine if pop's filling this job should be removed from further calculations and attempts to fill more jobs or not. Aka, will this job be non-exclusive and let pops double dip into another job as well.

I would prefer that script flag be capable of fractional deductions. 0.0 - 1.0
Or, if you want to allow more than one, or negatives, have at it. I'm sure someone will find a fun mod use for them.

--
I'm sure someone will logically make this into a part time employment or overemployment mod.

--

Though, maybe this non-exclusivity is a better way to model your 'resistance' jobs? People start off resisting somewhat in their free time. Or end up resisting with exclusively all of their time.

A planet still producing resources while still the problems of producing resistance creates more interesting choices. If the 'resistance' is so binary as just an off, on switch, the gameplay loop is shallow. If planet production = 0, then crush resistance = yes, is a very boring decision with no nuance. There's no drawback to crushing the resistance violently. If the planetary resource production is still up for negotiation, a wider range of options on the escalation ladder can credibly be applied.
 
  • 1Like
Reactions:
So for species traits, we encourage the use of these new modifiers

pop_job_bonus_workforce_mult
To increase the bonus workforce a pop generates for a given job.

pop_job_workforce_mult
To increase the workforce a pop generates for a given job, this is not bonus workforce.

job_max_workforce_mult
To increase the maximum workforce a Job can accept

As a reminder, a job's workforce will fill to its maximum allowed but not beyond that. If a pop generates more workforce than usual, fewer pops will be required to fill the job to max, but it will not produce more than its maximum. If a pop generates a bonus workforce, it can go beyond the job's maximum and scale its production up.
Two questions:

First- is job_max_workforce_mult something that can be scoped to the empire and planet level, not just applied to species? I can imagine, for example, some interesting applications of this with megacorp things. (A branch office building or a civic or a ruler trait or something.)

Second: are these designed to be able to apply to specific jobs *as well as* all jobs at once? I assume so, because even in species traits now there are +X% resources from jobs traits, but the way the post is written is just focusing on specific jobs.
 
Currently - not really. Right now we're still keeping it mostly single threaded and haven't gotten to many of the other possible optimizations yet, because it's a lot easier to understand what it's up to and debug this way.

Like I mentioned in another thread, we're likely to keep it this way during the early open beta so it's easier to fix any issues you guys hit, but I'm pretty confident about the new architecture being fundamentally better once we actually let it loose.
Question, IF it results that the performance improvements are not that great, is there any chance that trade routes (at least for now) stay in their current form?

More people have said it already, but essentially trade routes where the only 'resource-like' thing that needed some interaction from the player besides the usual loop of build-employ. You had to take piracy into account, trade collection etc. The resource didn't teleport to the capital, the system could be disrupted with war or even espionage (by being lucky and messing with the collection buildings/modules, which would be even better if espionage ever gets better). All in all, it was the only resource that was not 'just a number' even if later in the game it was gateways everywhere etc.

So, that, if the performance improvement results in being minimal, will we be keeping trade routes? I would assume yes, as they are being removed supposedly because of performance, if their removal doesn't improve it, then it should stay IMO. But I doubt that is the case.

Similarly, I must say this again, but I see no reason performance wise to make it that leaders not pick traits in each level up. And if the reason was annoyance wise, then there is an auto pick feature, not to mention that at the rate that leaders level up, compared with the plethora of things happening, they are pretty non disturbing IMO (again, compared to the rest of the game). At the end of the day, previously every level up felt good, the 'lets see what I can pick now' was nice, now you won't even notice some levels.

It will remove a lot from leaders (even if not directly power) for little gain. And that is completely ignoring that with less 'draws' (chances to pick traits) we are even more at the mercy of RNG, making leaders even worse (and they are already a pain to level except for most empires), ofc this last point would not me as much of an issue if leader traits where revised and balanced, but currently there are many that you want to avoid like the plague. Army starting experience, I am looking at you.
 
We mostly plan on using the non-bonus workforce modifiers for negative stuff, but for my example I'll also include a positive modifier. Imagine if there was a pair of traits Lazy (-50% Workforce) and Overachiever (+100% Workforce), and we have three species, Lazy Humans, Blorg, and Overachiever Starfish.

To fill 100 Miner jobs with each of the three species you would need:
  • 100 Blorg, they have no workforce modifiers so each Blorg contributes 1 Workforce.
  • 200 Lazy Humans, as each Lazy Human pop contributes 0.5 Workforce.
  • 50 Overachiever Starfish, as each Overachiever Starfish contributes 2 Workforce.
I agree that this not only is confusing, but it also might create scenarios where 'dumb' things happen. Specially when several of these effects are present at once. Pops that have non-bonus (from a building for instance) and bonus from a trait.

Could you provide an example of how would it work in that case?
 
  • 1Like
Reactions:
To simplify, lets get +100% bonus.
With workforce multiplier this means that a (for example) a machinist can work on two machines now. Since the amount of machines is limited, spare workers are laid off. Extended shifts edict is probably also this, since you need two workers working 12 hours each instead of 4 workers working 6 hours each.
With bonus workforce machines themselves work faster, like a psionic machinist can take material into machine and take product out without actually leaving the machine to go get them. So you don't have spare workers, you just have optimized workflow.
With job output you change the production method to give better results, so you get more products for the same materials.

Actually, what will happen to that three technologies that give capital buildings a bonus output? Will they produce bonus workforce now?
 
I think "+X% throughput to Farmer jobs" (and the like) would be more intuitive.

I can see the argument either way. On the one hand "Strong: +10% bonus workforce" is accurate to the game mechanics, but on the other it's not an intuitive explanation. If there were a bunch of labourers IRL moving rocks and one strong guy was doing it faster no one would say "his strength makes him generate so much bonus workforce!" More likely they'd say something along "his strength makes him more productive".

I'll see how it feels in the beta but tentatively I feel more natural language should be used with nested tooltips for mechanical explanations. E.g

Strong: +10% productivity to labourer and security jobs.

Then when the player hovers their mouse over "productivity" they get an explanation like "generates bonus workforce"
 
Last edited:
  • 6
  • 3Like
  • 1Love
Reactions:
We mostly plan on using the non-bonus workforce modifiers for negative stuff, but for my example I'll also include a positive modifier. Imagine if there was a pair of traits Lazy (-50% Workforce) and Overachiever (+100% Workforce), and we have three species, Lazy Humans, Blorg, and Overachiever Starfish.

To fill 100 Miner jobs with each of the three species you would need:
  • 100 Blorg, they have no workforce modifiers so each Blorg contributes 1 Workforce.
  • 200 Lazy Humans, as each Lazy Human pop contributes 0.5 Workforce.
  • 50 Overachiever Starfish, as each Overachiever Starfish contributes 2 Workforce.
Am I right that if I had Kinda Lazy (-10 wm) Agrarian (+10 bwm) humans working 100 farms would that result in 111.11 workers working the farms with a combined 11.111% increased upkeep and output? This is fine since it seems to be the balanced option as it results in you getting the same amount of extra food per pop, it's just interesting that they basically combo into a +11.111% worker capacity increase.
 
I hope the trait tooltips still just say "+10% food from jobs" because “+10% Bonus Workforce for Farmer Jobs” is a bit more wordy and honestly just kind of lame as a tooltip
"+10% Farmer Workforce"

I feel like "Bonus Workforce" should be renamed to "Throughput".

My read is there are two variables on workforce. One varies how much workforce a pop supplies, the other applies a bonus on top of the supplied workforce.

So one trait might make a pop supply double the workforce in a job. In which case, you only need 50 pops to fill a 100 workforce job. So you pay upkeep for 50 pops and 100 workfirce, getting 100 workforce output.

Another trait provides 10% bonus workforce. So in the same job, you need 100 pops but get 110 workforce output. That's upkeep for 100 pops and 110 workforce, with 110 output.
Oh, they are two values. Now it makes more sense.

pop_job_bonus_workforce_mult
To increase the bonus workforce a pop generates for a given job.

pop_job_workforce_mult
To increase the workforce a pop generates for a given job, this is not bonus workforce.

job_max_workforce_mult
To increase the maximum workforce a Job can accept
"Throughput" instead of "bonus workforce".

pop_job_throughput_mult
How much extra job Upkeep and yield the pop gives.

Way better term. You are probably only thinking about it in terms like "Network throughput", but it has usage in business:
Throughput in business is the rate at which a product is moved through a production process and onward to being consumed by an end-user[...]
Minerals to Alloys or Consumers goods is the definition of a Production Process.

It is a bit wierd with the basic resources, but mostly because you don't define a job upkeep. Just accept that as a oddity and focus more on the remaining 95% of jobs that come with job upkeeps.
 
  • 2Like
  • 1
Reactions: