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

Fix devlxd image export #13730

Merged
merged 12 commits into from
Jul 16, 2024
Merged

Conversation

markylaing
Copy link
Contributor

This was caused by calling CheckPermission without considering that the handler is called internally by devlxd.

In this PR I have updated all image handlers that have AllowUntrusted set to true. The handling of authentication/authorization is clarified with comments indicating what should happen when the caller is:

  • A guest instance.
  • An untrusted client.
  • An untrusted client with an image secret.
  • An authenticated client.

Comments contextual of whether the image is public, private, or cached.

Closes #13678
Closes #13167

@markylaing markylaing added the Bug Confirmed to be a bug label Jul 9, 2024
@markylaing markylaing requested a review from MusicDin July 9, 2024 15:55
@markylaing markylaing self-assigned this Jul 9, 2024
@markylaing markylaing requested a review from tomponline as a code owner July 9, 2024 15:55
@markylaing
Copy link
Contributor Author

@tomponline @MusicDin I'm guessing since devlxd image export was broken we don't have tests for it? I've tested it manually and from the speed of the image download in the guest it must have come from the host (the guest downloaded ubuntu:22.04 in about 3 seconds - I wish my internet connection was that fast 😆).

Another thing I wan't to be absolutely certain of is that an authenticated client can see public images in a project they don't have access to.

Shall I mark as draft for now while I work on tests? Otherwise I can put up a separate PR.

@MusicDin
Copy link
Member

MusicDin commented Jul 9, 2024

Thanks for this 🚀

I'm not sure how to reliably test it within CI. The instance needs internet access to fetch the fingerprint for the given image name/alias. Haven't tested, but maybe if fingerprint is provided instead of an alias, it will check for the image directly on the host. This would be handy as we can simply prevent instance's access to the internet and ensure image download does not fail.

Otherwise, I can think only of limiting the time for the image download to ensure it was fetched from the host (maybe this can be more reliable if we use some bigger images like Ubuntu).


For local development you can error out after this block:

devlxdHTTP, err := unixHTTPClient(nil, "/dev/lxd/sock", nil)
if err == nil {
resp, err := lxdDownloadImage(fingerprint, unixURI, r.httpUserAgent, devlxdHTTP.Do, req)
if err == nil {
return resp, nil
}
}

See the change in client/simplestream_images in this commit: 43fbdcc. This throws an error if image is not found on the host.


Also merging this PR should close this issue #13595

@markylaing
Copy link
Contributor Author

Thanks for this 🚀

I'm not sure how to reliably test it within CI. The instance needs internet access to fetch the fingerprint for the given image name/alias. Haven't tested, but maybe if fingerprint is provided instead of an alias, it will check for the image directly on the host. This would be handy as we can simply prevent instance's access to the internet and ensure image download does not fail.

Otherwise, I can think only of limiting the time for the image download to ensure it was fetched from the host (maybe this can be more reliable if we use some bigger images like Ubuntu).

For local development you can error out after this block:

devlxdHTTP, err := unixHTTPClient(nil, "/dev/lxd/sock", nil)
if err == nil {
resp, err := lxdDownloadImage(fingerprint, unixURI, r.httpUserAgent, devlxdHTTP.Do, req)
if err == nil {
return resp, nil
}
}

See the change in client/simplestream_images in this commit: 43fbdcc. This throws an error if image is not found on the host.

Also merging this PR should close this issue #13595

I think we should avoid calling out to external remotes if at all possible. I'll investigate whether it is possible to trick the guest LXD by using images:${test_image_full_fingerprint}. I'm hoping it will first see that the image is not cached, then call the devlxd endpoint with the full fingerprint without first trying to resolve the fingerprint as an alias. We'll see :)

@markylaing
Copy link
Contributor Author

I think we should avoid calling out to external remotes if at all possible. I'll investigate whether it is possible to trick the guest LXD by using images:${test_image_full_fingerprint}. I'm hoping it will first see that the image is not cached, then call the devlxd endpoint with the full fingerprint without first trying to resolve the fingerprint as an alias. We'll see :)

My last comment was a bit shortsighted 😆. We can't run nested LXD in the busy box test image. We'll have to put tests for this into the lxd-ci repo instead.

@tomponline
Copy link
Member

My last comment was a bit shortsighted 😆. We can't run nested LXD in the busy box test image. We'll have to put tests for this into the lxd-ci repo instead.

This is fine by me

@markylaing markylaing marked this pull request as draft July 10, 2024 12:58
@markylaing
Copy link
Contributor Author

Marking as draft for a bit. Still adding tests. I need to raise a couple of issues on the back of this as well.

@markylaing markylaing force-pushed the check-trusted-follow-up branch from b71f344 to 2426d4d Compare July 11, 2024 11:26
@markylaing markylaing marked this pull request as ready for review July 12, 2024 12:35
@markylaing
Copy link
Contributor Author

@tomponline this is ready for review. The associated test PR is here: canonical/lxd-ci#240

lxd/images.go Outdated Show resolved Hide resolved
lxd/images.go Outdated Show resolved Hide resolved
lxd/images.go Outdated
trusted := auth.IsTrusted(r.Context())

// Unauthenticated clients that do not provide a secret may only view public images.
// Devlxd queries can be for private images but only if they are cached.
Copy link
Member

Choose a reason for hiding this comment

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

Why are only private cached images allowed? I feel like this comment could do with expanding.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I've updated the comments here but I'm still not sure I've made it super clear to be honest.

Devlxd queries can be for public images because the image is accessible to anyone. Devlxd queries can be for private images only when the image is cached from a public remote. E.g. if the host has ubuntu:22.04 downloaded, we cache it but don't make it public.

Copy link
Member

Choose a reason for hiding this comment

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

Thanks, that makes sense.

We don't want to trigger a download of an image on the host.

BTW are we restricting which project images are accessible by /dev/lxd requests to only those of the instance's effective image project?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Currently are devlxd requests are for the default project.

I think this warrants a separate issue though. As I understand it, all cached images are stored in the default project. If I launch an ubuntu:22.04 container in project test say, and then create another ubuntu:22.04 inside it, then restricting the devlxd request to only project test would miss the cached image and download it via the ubuntu remote instead.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Another thing that doesn't make sense to me is how to actually make an image public on a host that is accessible to the nested LXD. The nested LXD doesn't have the host added as a remote, so it can't resolve the alias of the public image. As far as I can tell, this feature only works for cached images.

Copy link
Member

Choose a reason for hiding this comment

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

OK thanks, yes please can you create a separate issue.

I'm not sure that "all cached images are stored in the default project" but if /dev/lxd only servers from the default project currently then we dont need to change it in this PR.

Copy link
Member

@tomponline tomponline Jul 15, 2024

Choose a reason for hiding this comment

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

Another thing that doesn't make sense to me is how to actually make an image public on a host that is accessible to the nested LXD. The nested LXD doesn't have the host added as a remote, so it can't resolve the alias of the public image. As far as I can tell, this feature only works for cached images.

Does lxc image import server:fingerprint work?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

OK thanks, yes please can you create a separate issue.

I'm not sure that "all cached images are stored in the default project" but if /dev/lxd only servers from the default project currently then we dont need to change it in this PR.

#13768

Another thing that doesn't make sense to me is how to actually make an image public on a host that is accessible to the nested LXD. The nested LXD doesn't have the host added as a remote, so it can't resolve the alias of the public image. As far as I can tell, this feature only works for cached images.

Does lxc image import server:fingerprint work?

In canonical/lxd-ci#240 I experimented with creating an image on the host and making it public. This would prove that image export over devlxd was working because the image would not be present anywhere else. The problem is that you can't specify the host as a remote (e.g. what do you put as "server" in the command you posted), so I tried out a few things related to this comment. The methods I tried were:

  1. (naively) lxc init "ubuntu:{host_image_fingerprint}" c1. This doesn't work because the client tries to resolve the image against the ubuntu remote.
  2. Using lxc query -X POST /1.0/instances and providing an image source with the fingerprint of the public image on the host. For this you need to provide a remote URL, so I added the ubuntu URL. This fails because the guest LXD tries to get the image from the remote to perform architecture checks (here).
  3. Downloading the image directly via cURL. This doesn't really work because the downloaded octet-stream needs to be split into rootfs and metadata files before you can do anything. I also didn't like this because it doesn't actually test that the guest will download an image from the host when requested, only that the socket itself works.

@markylaing
Copy link
Contributor Author

Static analysis failing because I touched lxd/devlxd.go which has some pre-existing lint errors. Will fix shortly.

@markylaing markylaing force-pushed the check-trusted-follow-up branch from 931f1c1 to 31b5ffa Compare July 15, 2024 12:48
@markylaing
Copy link
Contributor Author

Static analysis failing because I touched lxd/devlxd.go which has some pre-existing lint errors. Will fix shortly.

Now fixed.

@markylaing markylaing marked this pull request as draft July 15, 2024 12:49
@markylaing markylaing force-pushed the check-trusted-follow-up branch from 31b5ffa to caf60bd Compare July 15, 2024 12:49
@markylaing markylaing marked this pull request as ready for review July 15, 2024 12:50
@markylaing markylaing force-pushed the check-trusted-follow-up branch from caf60bd to ebb2cbc Compare July 16, 2024 11:36
Go conventions state that single letter variable names should be used
for receivers, for familiar types (e.g. r for io.Reader or *http.Request)
or in very short scopes such as loops. They shouldn't be used for field
names. See
https://google.github.io/styleguide/go/decisions.html#single-letter-variable-names.

Signed-off-by: Mark Laing <[email protected]>
Field names are optional if private according to the Go style guide
(https://google.github.io/styleguide/go/decisions.html#field-names),
but for consistency in our code-base we should include the field names.

Signed-off-by: Mark Laing <[email protected]>
Additionally, if the `net.Conn` in `(*ConnPidMapper).ConnStateHandler`
is not a (*net.UnixConn), return early to avoid a panic.

Signed-off-by: Mark Laing <[email protected]>
Additionally standardise the error message.

Signed-off-by: Mark Laing <[email protected]>
@markylaing markylaing force-pushed the check-trusted-follow-up branch from ebb2cbc to f58a9cd Compare July 16, 2024 11:50
@markylaing
Copy link
Contributor Author

@tomponline I've clarified the commits to devlxd.go and addressed your other comments.

Copy link
Member

@tomponline tomponline left a comment

Choose a reason for hiding this comment

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

Thanks!

@tomponline tomponline merged commit 499f5d1 into canonical:main Jul 16, 2024
26 of 28 checks passed
tomponline added a commit to canonical/lxd-ci that referenced this pull request Jul 29, 2024
Adds tests for the devlxd socket in a container. These tests are based
on the devlxd tests for VMs, with additional testing for image export
over the socket.

Note: I wasn't sure where else to add this test other than in the github
workflow.

Also note: This test will fail on edge until
canonical/lxd#13730 is merged.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Bug Confirmed to be a bug
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Auth: Follow up on removal of checkTrustedClient
3 participants