Skip to content
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

[release/1.7] Add a build tag to disable std plugin import (#11202) #11203

Merged

Conversation

pgimalac
Copy link

@pgimalac pgimalac commented Dec 27, 2024

What does this PR do?

Add a no_dynamic_plugins build tag in plugin/plugin_go18.go to allow disabling the plugin import, without using static_build (which is used by other dependencies).

Motivation

See #11202.

Importing plugin makes binaries significantly bigger due to preventing some linker optimizations, I would like to avoid that.
Removing the import causes a 31% reduction in binary size on one binary I work on (PR).

@k8s-ci-robot
Copy link

Hi @pgimalac. Thanks for your PR.

I'm waiting for a containerd member to verify that this patch is reasonable to test. If it is, they should reply with /ok-to-test on its own line. Until that is done, I will not automatically test new commits in this PR, but the usual testing commands by org members will still work. Regular contributors should join the org to skip this step.

Once the patch is verified, the new status will be reflected by the ok-to-test label.

I understand the commands that are listed here.

Instructions for interacting with me using PR comments are available here. If you have questions or suggestions related to my behavior, please file an issue against the kubernetes-sigs/prow repository.

@dosubot dosubot bot added the area/toolchain Build and Release Toolchain label Dec 27, 2024
@pgimalac pgimalac force-pushed the pgimalac/containerd-no-plugin branch from c29d974 to 0ef0f3f Compare December 27, 2024 15:16
@thaJeztah
Copy link
Member

Honestly, I wonder if anyone is actually using this and if it should've been deprecated for v2.0. I think the hopes were to allow this to be used, but their usefulness is very limited, and I think Go itself largely abandoned any effort to make it better (I don't think it's even supported on all platforms).

Perhaps it could even be considered to do the reverse, and make it "opt-in".

Copy link
Member

@AkihiroSuda AkihiroSuda left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@AkihiroSuda
Copy link
Member

@thaJeztah
Copy link
Member

This is actually deprecated in https://github.com/containerd/containerd/blob/v2.0.1/RELEASES.md#deprecated-features

Ah! Nice. Perhaps it would make sense to include this in existing release branches to already allow packagers to disable it (and then remove from main)

@pgimalac pgimalac force-pushed the pgimalac/containerd-no-plugin branch from c38d849 to 6092b9b Compare December 28, 2024 15:32
@pgimalac
Copy link
Author

Please squash the commits and update main/BUILDING.md#build-containerd

Done 👍
I wasn't sure how detailed the build tag description in BUILDING.md should be so I kept it (very) short, let me know if you want me to change it !

Note that this PR targets release/1.7 directly, not the main branch.
If you want the build tag in v2 then I believe I would have to open a similar PR in github.com/containerd/plugin.

@AkihiroSuda AkihiroSuda changed the title Add a build tag to disable std plugin import (#11202) [release/1.7] Add a build tag to disable std plugin import (#11202) Dec 28, 2024
@AkihiroSuda AkihiroSuda added the cherry-pick/1.6.x Change to be cherry picked to release/1.6 branch label Dec 28, 2024
@pgimalac
Copy link
Author

I checked out of curiosity, using the build tag reduces the size of containerd itself from ~56MB to ~43MB, a 24% reduction 😄

@pgimalac pgimalac force-pushed the pgimalac/containerd-no-plugin branch from 6092b9b to 2b12ef2 Compare December 28, 2024 20:36
@pgimalac
Copy link
Author

👋 Hello @AkihiroSuda @djdongjin @thaJeztah, is there anything you want me to change, or anything I can do to get this merged ?

@thaJeztah
Copy link
Member

Note that this PR targets release/1.7 directly, not the main branch.
If you want the build tag in v2 then I believe I would have to open a similar PR in github.com/containerd/plugin.

Yes, we would probably need that, so that builders can opt-out of this feature both for 1.7 and 2.0 until it's removed.

Perhaps the alternative would be to handle this on the containerd side; that way we don't need a build-tag in containerd/plugins; that module doesn't necessarily have to be updated because importing the dynamic package is not enforced by the module, so we can leave it up to the consumer (containerd in this case) to decide whether to use it.

It looks like the dynamic loading happens here;

func LoadPlugins(ctx context.Context, config *srvconfig.Config) ([]plugin.Registration, error) {
// load all plugins into containerd
path := config.PluginDir //nolint:staticcheck
if path == "" {
path = filepath.Join(config.Root, "plugins")
}
if count, err := dynamic.Load(path); err != nil {
return nil, err
} else if count > 0 || config.PluginDir != "" { //nolint:staticcheck

So we can stub that out based on build-tag, e.g. having a plugins.go and plugins_nodynamic.go;

//go:build no_dynamic_plugins

package server

func loadDynamic(path string) (loaded int, _ error) {
	// dynamic plugins disabled through no_dynamic_plugins build-tag
	return 0, nil
}

And change the server code to call that instead;

func LoadPlugins(ctx context.Context, config *srvconfig.Config) ([]plugin.Registration, error) {
	// load all plugins into containerd
	path := config.PluginDir //nolint:staticcheck
	if path == "" {
		path = filepath.Join(config.Root, "plugins")
	}
	if count, err := loadDynamic(path); err != nil {

Or we could even move out the whole bit about loading the dynamic ones;

func loadDynamic(ctx context.Context, config *srvconfig.Config) error {
	path := config.PluginDir //nolint:staticcheck
	if path == "" {
		path = filepath.Join(config.Root, "plugins")
	}
	count, err := dynamic.Load(path)
	if err != nil {
		return err
	}
	if count > 0 || config.PluginDir != "" { //nolint:staticcheck
		config.PluginDir = path //nolint:staticcheck
		log.G(ctx).Warningf("loaded %d dynamic plugins. `go_plugin` is deprecated, please use `external plugins` instead", count)
	}
	return nil
}

and stub;

func loadDynamic(ctx context.Context, config *srvconfig.Config) error {
	// dynamic plugins disabled through no_dynamic_plugins build-tag
	return nil
}

Copy link
Member

@thaJeztah thaJeztah left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Saw a minor issue in formatting. Changes otherwise LGTM for 1.7

Wondering if we should get the suggested changes for 2.0 in first (to make sure we don't "regress" and support something for 1.7, but don't support it on 2.0).

@pgimalac
Copy link
Author

pgimalac commented Dec 31, 2024

I opened a PR with similar changes targeting release/2.0, let me know if there is anything else I can do.
#11213

Copy link
Member

@thaJeztah thaJeztah left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM

@pgimalac
Copy link
Author

pgimalac commented Jan 6, 2025

PR targeting release/2.0 is merged, any chance you can also re-review this one @AkihiroSuda ?

@AkihiroSuda AkihiroSuda merged commit 142e855 into containerd:release/1.7 Jan 6, 2025
56 of 57 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
area/toolchain Build and Release Toolchain cherry-pick/1.6.x Change to be cherry picked to release/1.6 branch ok-to-test size/XS
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants