-
-
Notifications
You must be signed in to change notification settings - Fork 21.5k
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
GDScript: Reintroduce binary tokenization on export #87634
Conversation
d782894
to
c5a5728
Compare
But what happened with intermediate representation proposal, is there is any work done there? Why bring back this useless tokenization if there is zero protection to the source code because gdsdecomp can recover everything in a second? |
Would be great to see exported projects size comparison |
To be clear, this is unrelated to the IR proposal. The proposal itself will take some time to solidify because it needs approval from the maintainers and not everyone think it's feasible. This PR is a palliative to avoid having plain source code export while we wait for that or figure out something else, because a lot of people has been complaining about this issue. |
c5a5728
to
88698e1
Compare
But returning tokenization does not solve this problem because scripts are still easily opened with gdsdecomp tool. |
That won't change in any real way with any solution, any compiled/parsed/processed format will be vulnerable to that, it's just a matter of degrees, the biggest improvement will be this over plain source to the vast majority of "attacks" |
You can't run decompiled code because it's gibberish but in the case of GDScript tokenization you absolutely can revert it back to it's original form and run it without any issues, this is why this tokenization is useless. |
I'd suggest reading up on this and getting a realistic perspective on this, if you genuinely believe that decompilation isn't a realistic way of reverse engineering code then you are not going to have realistic expectations of the outcomes of these things
Please be realistic about the time it takes to undertake such a major project, and the explanations for the expectations on it above by vnen... Just because it hasn't happened yet doesn't mean it won't, especially with just a few months If you'd rather wait longer with no changes and no improvements for performance then that's your opinion, but I don't think it's reasonable to wait for a feature that doesn't clash with this just to get a bit better obfuscation (not that obfuscation is the main goal here anyway) (Also if you're going to block me you will not react to my comments please, that's just petty, I don't want to have to block you back...) |
Reducing loading times and applying zstd compression is good reason to reintroduce binary tokenization on its own merits. I don't know if I'll have much success reviewing the code. |
editor/export/editor_export_preset.h
Outdated
@@ -84,6 +90,7 @@ class EditorExportPreset : public RefCounted { | |||
bool enc_directory = false; | |||
|
|||
String script_key; | |||
int script_mode = MODE_SCRIPT_BINARY_TOKENS; |
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 think compressed is probably better default.
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.
Looks good on my end.
@MichaelWengren you can decompile and modify anything, be it .net or anything else. |
88698e1
to
ac3fe4d
Compare
Updated to set the compressed mode as the default. |
// Script export parameters. | ||
|
||
VBoxContainer *script_vb = memnew(VBoxContainer); | ||
script_vb->set_name(TTR("Scripts")); | ||
|
||
script_mode = memnew(OptionButton); | ||
script_vb->add_margin_child(TTR("GDScript Export Mode:"), script_mode); | ||
script_mode->add_item(TTR("Text (easier debugging)"), (int)EditorExportPreset::MODE_SCRIPT_TEXT); | ||
script_mode->add_item(TTR("Binary tokens (faster loading)"), (int)EditorExportPreset::MODE_SCRIPT_BINARY_TOKENS); | ||
script_mode->add_item(TTR("Compressed binary tokens (smaller files)"), (int)EditorExportPreset::MODE_SCRIPT_BINARY_TOKENS_COMPRESSED); | ||
script_mode->connect("item_selected", callable_mp(this, &ProjectExportDialog::_script_export_mode_changed)); | ||
|
||
sections->add_child(script_vb); |
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.
Technically this is GDScript specific code spilling into the main editor, which wouldn't be relevant if the GDScript module is disabled. So it's slightly breaking encapsulation.
But it's not a first, and I'm not sure it's worth adding a complex abstraction for this so it can be handled in the module for each script language individually. Your call if you think it's worth looking into.
For now I think this doesn't need to prevent merging.
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 though about this but I'm not sure of what would be the best solution. This is the same way it is handled in 3.x.
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.
Some accidental reverts, see above
I thought |
This adds back a function available in 3.x: exporting the GDScript files in a binary form by converting the tokens recognized by the tokenizer into a data format. It is enabled by default on export but can be manually disabled. The format helps with loading times since, the tokens are easily reconstructed, and with hiding the source code, since recovering it would require a specialized tool. Code comments are not stored in this format. The `--test` command can also include a `--use-binary-tokens` flag which will run the GDScript tests with the binary format instead of the regular source code by converting them in-memory before the test runs.
Besides the regular option to export GDScript as binary tokens, this also includes a compression option on top of it. The binary format needs to encode some information which generally makes it bigger than the source text. This option reduces that difference by using Zstandard compression on the buffer.
ac3fe4d
to
72e5f8c
Compare
Thanks! |
Just wanted to say I really appreciate this commit. I don't see obscuring the code as a security feature, but making the files smaller and faster is great. Most importantly, having comments stripped is very much a value add for a lot of developers. It's easy to leak things accidentally there since many devs used to compiled languages would assume comments are not user-visible. |
about source safety (reverse-engineering protection, obfuscation, etc) there's any thread to follow? |
@azur-wolf See: There are also proposals that can make decompilation more difficult, although they are not directly intended for this: |
Hi @vnen : thank you for all the hard work you do. Can I ask if this work is replacing variable names? While classes and function names would also be great to obfuscate, the number one thing to minify would be variable names. Many software organizations don’t write much in the way of comments, and symbols go a long way towards being the documentation. Having the tokenizer scramble variable names would really take a huge step towards making the intent of a function opaque. |
No, it keeps identifiers as-is. Local variables could technically be renamed, but this is nontrivial to implement. Member variables can't be renamed without breaking the API that other classes may attempt to use, especially since there is no way to mark member variables as private yet. |
Without some sort of string name server that's able to recognise internal/GDExtension names for the entire project, not just inside single GDScript, it's impossible to alternate source code safely and makes external API access attempts impossible. There are also dire limitations for this approach such as inability to recogise fields inside string syntaxes. Take a look at this snippet. extends ColorRect
class_name MyColorRect
var my_own_custom_field := 0.0
func _ready() -> void:
var my_tween := create_tween()
my_tween.tween_property(self, "color", Color.BLACK, 1.0)
my_tween.tween_property(self, "my_own_custom_field", 1.0, 1.0) If source code is alternated somehow and get decompiled, it may become something like this. extends ColorRect
class_name _z0
var _z1 := 0.0
func _ready() -> void:
var _0 := create_tween()
_0.tween_property(self, "color", Color.BLACK, 1.0)
_0.tween_property(self, "my_own_custom_field", 1.0, 1.0) You'll notice immediately that While it's not difficult to overcome the issue by just deep-scanning strings and replace them accordingly (and it will become extends Node
class_name WorldController
@onready var world := get_tree().root
func _ready() -> void:
print("Hello, world!") It may become: extends Node
class_name _z0
@onready var _z1 := get_tree().root
func _ready() -> void:
print("Hello, _z1!") Instead of it printing There are other projects that cover this type of source mangling approach such as GDMaim, but you still need to be careful while utilising the tool. |
This adds back a function available in 3.x: exporting the GDScript files in a binary form by converting the tokens recognized by the tokenizer into a data format.
It is enabled by default on export but can be manually disabled. The format helps with loading times since, the tokens are easily reconstructed, and with hiding the source code, since recovering it would require a specialized tool. Code comments are not stored in this format.
The
--test
command can also include a--use-binary-tokens
flag which will run the GDScript tests with the binary format instead of the regular source code by converting them in-memory before the test runs.Besides the regular option to export GDScript as binary tokens, this also includes a compression option on top of it. The binary format needs to encode some information which generally makes it bigger than the source text. This option reduces that difference by using Zstandard compression on the buffer.