-
Notifications
You must be signed in to change notification settings - Fork 149
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fixes for WorldSaveData dimension handling (#1398) and fluid in pipes… #1399
base: master
Are you sure you want to change the base?
Conversation
…d in pipes not being persisted (GregTechCE#1357)
Thank you for creating this PR. These are some serious claims regarding problems in our WorldPipeNet I will need to look at this myself as I am not currently able to tell if it is correct. Obviously I am quite surprised because we did not have any reports regarding this. And fix which will break someone bases is not something I can easily allow in mod. If anyone come in contact with something which could be relate to problems which this fixes please step up and let me know. |
Did you try the simple test on the bug report? For the central issue, look at WorldServerMulti which is the thing that handles the non-overworld dimensions. In WorldServerMulti.init() you will find
mapStorage is where the pipenets get stored
|
To try to avoid breaking old data, I was thinking we could do something to try detect old data and then turn off the bug fix. This could be something like changing the overworld data to have .0 on the end, in line with how the other dimensions are saved. If we can find data with the old data ids, we leave the bug in place. It's not ideal, but since people have been happily playing with this bug in place they probably wouldn't notice it still being there. The could also manually enable the bug fix for an old world if they have only created pipenets in the overworld i.e. in saved-game-name/data/ change gregtech-e_net.dat (old name) to gregtech-e.net.0.dat (new name) and the same This patch still uses the old name for the overworld to try to migrate data. |
…ists for backwards compatibility
Thank you for providing compatibility for current worlds this will make it more likely to get accepted. Regarding test on my end I was not yet able to do it as my current dev environment is broken. |
Is this possibly related to #1256? I understand that the issue is showing itself from a different part of the PipeNet, being |
I don't think it is related. The first crash doesn't involve pipes and the second is in the cover processing of a pipe which is "duplicate code" of the metatileentity processing. The pipenet code typically checks for World.isBlockLoaded() before accessing tiles so it can't cause cascades. |
I finally get to this and tested your claims and implementation. As you pointed out there are problems with with WorldPipeNet. And your analysis is up to point and it needs to be fixed. When testing on new world there were no problems found. When testing conversion from before this updated I have confirmed that pipenet does not get broken in all dimensions (fluids still go through system without need to replace pipes). But I have found out that wooden pipes when transporting steam in nether still does not burn. Can you please have a look at this case? |
For the pipes don't burn in the nether. If you have old data, yes that will still be a problem. The way to activate the bug fix for old saves is to go into saves/world-name/data and rename gregtech.pipenet-name.dat to gregtech.pipenet-name.0.dat and then do the break and replace on pipes not in the overworld. |
Can't we just discard old data entirely and lazily rebuild it once any of the emitter blocks requires it? |
No, because the saved data remembers any manual disconnections that were made with a wrench. |
If we could rebuild the data lazily, there would be no need to save the data at all. |
Incorrect, the point of saving data is to avoid world access and chunk loading while doing net-related operations. Other than that, there is no point in saving that data at all.
Incorrect again. Wrench disconnections are stored inside of the pipe tile entities, and are simply mirrored into the saved data. They can be easily reconstructed from just tile entity data. Overall, I see nothing stopping us from implementing lazy data regeneration. |
Ok, so you can in principle regenerate the node data from the pipes. But doesn't that mean forced chunk loading to do recalcuations of things like cable route paths or the total fluid capacity of the fluid pipes when a pipe is loaded? And doing this everytime a pipe is loaded (or maybe there is a way to do this only if you detect it's node data is missing?). You may also need to cascade into neighbouring tiles as well in case they have things like shutters or to check their capabilities. My statements above assumed that you don't want this to happen, see my comments on #1256 about doing something like the pipenet optimisations for the redstone signals in tiles and covers. Also, if the philosophy is that the pipe data is just a cache, my fix for storing the fluid data there is wrong, since it is the actual data. |
Yeah, you will definitely need to force load chunks and cascade into neighbour tiles to regenerate the data. But again, you only need to do it once and only if data in question is missing. Design philosophy that pipe net is just a data cache is not entirely right. It is the case for cables, but for pipes network in fact owns the currently stored fluid. On the other side, loss of stored fluid is not that critical because internal buffer of pipes is rather small. Alternative solution would be saving/loading old data as-is, but once network element requires access to the pipe net owning given position, pipe net in question is removed from old data and moved into the new data associated with the world original requester pipe is in. That way fix can be applied to old world's without completely regenerating their pipe nets. |
I am not saying this is not doable, but the purpose of this is to try to fix old data. A hackier/easier alternative is to just copy the old data into every dimension. But this runs the risk that at some point a blockpos will overlap with old data from a different dimension. |
We seem to have crossed and have had the same idea? |
No, my approach involves keeping old data shared with all dimensions, and then moving it once owner dimension is found. The difference is that if you just copy the data for all dimensions, it will remain dangling for all the dimensions except needed one, and if you have a shared old data, you can clear it once ownership has been claimed by one of the worlds. |
ok, i will give that a try and see what problems I find. The most obvious one is where there is already a conflicting blockpos in the data. |
Ok, I have committed a fix that automatically moves the old data into the new data as Archengius suggested. This looks to be working except for the overlapping blockpos which I will address in a seperate comment.
On implementation details: There are additional safety checks before it fixes data. |
For the overlapping blockpos of pipenets in different dimensions this is worse than I thought. I had thought the getNetFromPos() would choose a random pipenet based on how it processes the list of pipenets in a chunk. But from my testing, this doesn't seem to happen in practice. Instead, the pipenets will get merged when you create them. Worse than that, it is not just the same blockpos, it is neighbouring blocks as well, because of the way addNode() checks the block connections to the surrounding blocks. If this has already happened in somebody's world, the data is broken and unfixable. They are probably already experiencing buggy behaviour? I have however, placed a warning in the code in case it detects such duplicate data. It won't attempt to fix it. |
Actually the "It won't attempt to fix it" is only half true. If there is another pipe in the same pipenet that does not overlap with a pipe in a different dimension, that is enough for the basic data fix to work. |
I understand gist of last update to this PR. But I am not confident enough to review if this approach is correct. So please @Archengius can you look through this. |
I believe quite confidently now that these two issues are related. The CME crash caused in this issue only occurs when the ID of a material with CableProperties is changed for a save. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Other than multiple minor formatting/codestyle issues, there is a core issue in the way you are doing the conversion of the old data. I've covered it by the comment on onLoaded. Tl;dr: it's a pretty hot spot and trying to convert data here is probably a bad idea. Alternatives have been suggested in my comment there too.
@@ -27,16 +27,25 @@ public static void registerTickablePipeNet(Function<World, TickableWorldPipeNet< | |||
|
|||
@SubscribeEvent | |||
public static void onWorldTick(WorldTickEvent event) { | |||
getPipeNetsForWorld(event.world).forEach(TickableWorldPipeNet::update); | |||
final World world = event.world; | |||
if (world == null || world.isRemote) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You shouldn't check world for null here, these events simply should never have it null.
Might be a good idea to refactor if-not-return to if-then statements, too.
} | ||
|
||
@SubscribeEvent | ||
public static void onChunkLoad(ChunkEvent.Load event) { | ||
getPipeNetsForWorld(event.getWorld()).forEach(it -> it.onChunkLoaded(event.getChunk())); | ||
final World world = event.getWorld(); | ||
if (world == null || world.isRemote) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ditto.
} | ||
|
||
@SubscribeEvent | ||
public static void onChunkUnload(ChunkEvent.Unload event) { | ||
getPipeNetsForWorld(event.getWorld()).forEach(it -> it.onChunkUnloaded(event.getChunk())); | ||
final World world = event.getWorld(); | ||
if (world == null || world.isRemote) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same thing again.
if (foundOldData != null) | ||
{ | ||
// We have 2 pipenets at this position? | ||
GTLog.logger.warn("Found duplicate pipenets in old data at " + blockPos + " [" + foundOldData + ',' + pipeNet + ']'); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You should use logging formatting instead of string concatenation here.
if (foundOldData == null) | ||
return; | ||
// Move the old data into the new data | ||
GTLog.logger.info("Fixing old data for " + foundOldData + " found at " + blockPos); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Same thing.
super.deserializeNBT(nbt); | ||
final NBTTagCompound tankData = nbt.getCompoundTag("FluidNetTank"); | ||
// Guard against old data that didn't have this tag | ||
if (tankData != null) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Use if (nbt.hasTag("FluidNetTank")) here instead.
final World world = getWorld(); | ||
final BlockPipe<PipeType, NodeDataType, ?> pipeBlock = getPipeBlock(); | ||
if (world != null && !world.isRemote && pipeBlock != null) { | ||
pipeBlock.getWorldPipeNet(world).checkForOldData(getPos()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not sure if this is a good place to check for legacy data. Since it will be called pretty often (actually for every pipe-like tile entity loaded), it might incur some overhead, especially taking into account that usually pipe tile entities don't perform any work themselves and don't even tick unless they have to.
Besides, I'm not really sure if you can obtain the block object in question at that point. You probably can, though, because it's saved in NBT. Although it was not always the case, but I think we shouldn't care too much about ancient versions where pipe block was actually retrieved from the world object and never saved.
It might be a good idea to move these checks to some kind of place which is not called that often, or at least not for all pipelike tile entities in the world. Some places that come to mind quickly: "active" fluid pipe tile entities, "source" type cable tile entities, onBlockAdded which checks for neighbor networks.
Although it leaves logic split across multiple places, so it's not the best solution there.
Alternatively we can try to move migration logic to getNetFromPos, e.g. if pipe net is not found, it will try to check for it by digging through old data and restoring it from there. Since getNetFromPos is mostly used only in 2 cases (performing actual transfer logic and joining neighbor nets), and on case 1, which is also the most common one, it will always have a valid network, it shouldn't incur any kind of noticeable performance impact. On case 2 it shouldn't really matter, since that logic is only caused by players placing/breaking blocks, and these actions just don't happen every tick, and even then, hash map lookup is pretty fast.
I personally think last suggestion is the best one to implement, but I'm always open for a civil discussion, especially from @LAGIdiot.
I have done the changes you asked for, with one exception. getCompoundTag() already does the hasKey() test with the additional check that it is actually a compound tag (type 10). The only reason for adding this check is so we don't run into an NPE with old data. |
On the main issue, I can see that putting the check for old data at the end of getNetPos() is slightly more efficient. Most of the calls are from network topology changes. These are only obtained for the "active" pipes as you say, and they are cached after their first use. It does however mean that non-active pipes don't have to do this processing. If as I mentioned above, one of these pipes could be used to resolve a pipe conflict across dimensions, it won't happen. As an aside. I think the FluidPipeFluidHandler could actually be resolved once and shared between all fluid pipes in the pipe network? |
But maybe that doesn't save you very much since you still need to go through getNetFromPos() at least once. |
I don't think sharing fluid handler will save much. Honestly though, both cables and pipes ideally should be refactored to use network segments to make their behaviour more consistent. |
On getCompundTag(). I can see both, even in the same code, e.g. MetaTileEntity :-)
But it is difficult to find anything that is doing the same as this usecase. I don't think "everybody else is doing it" is a good excuse for doing it the wrong thing (or in this case the inefficient thing). :-) Same with those changes you asked me to make for logging. |
About the getTagCompound, here is the very simple difference between the code you provided and what we have in this PR: getTagCompound in the provided code is called on ItemStack, since stacks can have no component at all and it's a pretty normal thing. In the code of PR (and the cases I mentioned) we are talking about retrieving children tags from the NBTTagComponent. Since numbers and other stuff are technically subtags too but they cannot have null value, we are using hasTag to check for their existence before trying to use them, because otherwise we would end up with 0 if the tag doesn't exist. So to keep things uniform, you should use hasTag instead of checking getCompoundTag result for null. And regarding logging, no, string concat is not readable at all, that's why people invented formatting and built up logging libraries that support it. It's much easier to see the original message in one place without having it split by the values inside. |
I did your change and while I was testing it I found real problem Basically the FluidNetTank was not marking the pipenet as dirty when the fluid amount changes. I can't believe I hadn't spotted it before, but I guess the way I was testing it meant the pipenet was always getting marked as dirty due to connection changes when I broke tanks. Anyway, I will do some more testing on this tomorrow or later to see if I can find another way to break it. |
And that's why you don't make nonfunctional changes, it introduces unintended bugs.
|
@Archengius Can you please give this another look. There were some changes from your last comment. |
This reverts commit fe74783.
… not being persisted (#1357)
What:
I have included the fix for #1357 (fluid in pipes is not persisted across restarts/reloads) because I found the main issue while testing that fix.
The main fix is for #1398 which is about properly separating pipenets by dimension and not running pipenet processing on the client.
How solved:
Data for dimensions other than the overworld now have their dimension number appended to the WorldSaveData data id.
e.g. gregtech.fluid_pipe_net is used for the overworld and gregtech.fluid_pipe_net.-1 is used for the nether.
The reference to the world in the WorldPipeNet has been changed to a WeakReference so it doesn't stop garbage collection
of an unloaded dimension, and setWorldInit() can now handle reloading of dimensions by checking for a change instead
of the firstTick flag.
Additionally there are corrections to stop pipenet processing running on the client.
With an additional check to try to spot this happening in future changes.
Outcome:
Fixes: #1398, correctly separate pipenets by dimension.
Fixes: #1357 persist fluid in pipes across reloads.
Possible compatibility issue:
This is the elephant in the room.
People that have built bases in other dimensions, e.g. in space in OmniFactory or FTB interations will have their pipenets currently saved against the overworld.
Despite that it would have kind of worked provided no cables/pipes have the same blockpos in the 2 dimensions.
e.g. PipeNet.getNetFromPos() would choose a "random" dimension's pipenet if block positions are shared
This fix looks for non-overworld data somewhere else, so old pipenet data for non-overworld dimensions will not be used.
There is no way to migrate this old data. The necessary dimension info is not available in the old data to fix it.
To fix these old non-overworld cable/fluid networks, the blocks will need to be broken and replaced to recreate them.
Please fill in as much useful information as possible. Also please remove all unused sections.