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

More Mount Conflict Detection #2919

Merged
merged 17 commits into from
Nov 6, 2017
Merged

Conversation

otakup0pe
Copy link
Contributor

@otakup0pe otakup0pe commented Jun 24, 2017

Ensure that we are not creating a "submount" underneath an already
existing mountpoint. This (hopefully) fixes #2878

Note that (locally) I'm seeing intermittent (~25% of the time on the full test suite) build failures 😿

--- FAIL: TestCluster_ForwardRequests (9.16s)
        cluster_test.go:350: err: error during forwarding RPC request

Ensure that we are not creating a "submount" underneath an already
existing mountpoint. This (hopefully) fixes hashicorp#2878
@otakup0pe
Copy link
Contributor Author

I'm missing something about the intersection between the router and logical backends... Doing some (manual) smoke testing, I noticed this repeating in the logs...

2017/06/24 10:14:45.630510 [ERROR] rollback: error rolling back: path=foo/ error=unsupported path

@jefferai
Copy link
Member

The issue is that what's in the router's Mount code is only an extra check; the code in mount.go and auth.go check for an exact match via the router's MatchingMount, but then if that doesn't find a conflict they add the new entry into the table, then mount it on the router. The reason is that once a backend is able to be sent requests, we want it to already be stored so if Vault goes down it can come back up.

The right thing to do would probably be to take the logic you wrote and the check above for a longest prefix and combine it into two functions inside the router, something like AllowedMount and allowedMountInternal. The latter takes that logic and returns an error or not, and can be used inside the router's Mount function. The former does the same thing but gets a read lock (see MatchingMount); this can be called from mount.go and auth.go instead of the existing MatchingMount call.

@jefferai jefferai added this to the 0.7.4 milestone Jun 24, 2017
@otakup0pe
Copy link
Contributor Author

otakup0pe commented Jun 25, 2017

Thanks for the guidance, those changes cleared up the rollback error. Any guidance on why (local) postgres tests are (consistently) failing? Basic test suite (make test TEST=./vault) is fine.

--- FAIL: TestPostgreSQL_RenewUser (8.30s)
        postgresql_test.go:182: Could not connect with new credentials: pq: password authentication failed for user "v-test-test-ZLS5aFHTd2RNsz8eDKV8-1498407266"
--- FAIL: TestPostgreSQL_RevokeUser (7.56s)
        postgresql_test.go:250: Could not connect with new credentials: pq: password authentication failed for user "v-test-test-WzseS2eTjqgckXSBWefZ-1498407274"
FAIL
FAIL    github.com/hashicorp/vault/plugins/database/postgresql  36.679s

vault/auth.go Outdated
@@ -73,7 +73,7 @@ func (c *Core) enableCredential(entry *MountEntry) error {
return fmt.Errorf("token credential backend cannot be instantiated")
}

if match := c.router.MatchingMount(credentialRoutePrefix + entry.Path); match != "" {
if match := c.router.MountConflict(credentialRoutePrefix + entry.Path); match != "" {
Copy link
Member

Choose a reason for hiding this comment

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

For ease of understanding, can you change match here to conflict and in the error message say "conflicting mount at ..."?

vault/router.go Outdated
// with an existing mountpoint
if prefix != "" {
if match := r.MatchingPrefix(prefix); match != "" {
return fmt.Errorf("cannot mount under existing mount '%s'", match)
Copy link
Member

Choose a reason for hiding this comment

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

Isn't this checking for mounting over an existing mount?

vault/router.go Outdated
func (r *Router) MatchingPrefix(path string) string {
var existing string = ""
fn := func(existing_path string, _v interface{}) bool {
fmt.Println(fmt.Sprintf("comparing %v %v", path, existing_path))
Copy link
Member

Choose a reason for hiding this comment

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

I'm guessing this was a debug line you meant to take out

Copy link
Contributor Author

Choose a reason for hiding this comment

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

How embarrassing 😬

Copy link
Member

Choose a reason for hiding this comment

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

Mostly it was like, um, that's going to spam logs a LOT :-)

Copy link
Contributor Author

@otakup0pe otakup0pe Jun 25, 2017

Choose a reason for hiding this comment

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

Which is why I'm feeling a touch embarrassed 👼

New commit incorporates the feedback and I added a little blurb to the docs about mount conflicts.

vault/router.go Outdated
@@ -174,6 +181,31 @@ func (r *Router) MatchingMount(path string) string {
return mount
}

//MatchingPrefix returns a mount prefix that a path may be a part of
func (r *Router) MatchingPrefix(path string) string {
Copy link
Member

Choose a reason for hiding this comment

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

This function doesn't get a read lock but should, as it can be called externally (via MountConflict from the mount code). However, it is called with a lock already held from the router's Mount function.

The way to deal with this would be to have MatchingPrefix grab a read lock, then call matchingPrefixInternal which has the actual logic. The router's Mount function would then just call matchingPrefixInternal directly.

@@ -59,6 +59,13 @@ Once a secret backend is mounted, you can interact with it directly
at its mount point according to its own API. You can use the `vault path-help`
system to determine the paths it responds to.

Note that mount point cannot conflict with each other in Vault. There are
Copy link
Member

Choose a reason for hiding this comment

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

s/point/points

vault/router.go Outdated
//MatchingPrefix returns a mount prefix that a path may be a part of
func (r *Router) MatchingPrefix(path string) string {
var existing string = ""
fn := func(existing_path string, _v interface{}) bool {
Copy link
Member

Choose a reason for hiding this comment

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

If this function executes at all wouldn't it mean that we've found a prefix? Is the if check unnecessary here?

@otakup0pe
Copy link
Contributor Author

Added the read lock and fixed a typo but wasn't clear on the third statement. The if in what is now matchingPrefixInternal checks each entry in the radix tree... If you meant the if on line 67 - that is there because some entities (like policies iirc) didn't seem to have a prefix set...

@otakup0pe
Copy link
Contributor Author

otakup0pe commented Jul 11, 2017

Just noticed tests failed (they are passing locally). Will take a look this afternoon/evening... Unless someone can restart the build 👼

I see the build is passing now, but I'm seeing the following error consistently locally...

--- FAIL: TestPostgreSQL_RenewUser (8.98s)
	postgresql_test.go:182: Could not connect with new credentials: pq: password authentication failed for user "v-test-test-PyffKa0TNLrdv8zX1sGc-1499798498"
--- FAIL: TestPostgreSQL_RevokeUser (7.84s)
	postgresql_test.go:250: Could not connect with new credentials: pq: password authentication failed for user "v-test-test-9c8dmq3SMXLPFhYK2q4Z-1499798506"

vault/router.go Outdated
@@ -62,6 +62,13 @@ func (r *Router) Mount(backend logical.Backend, prefix string, mountEntry *Mount
if existing, _, ok := r.root.LongestPrefix(prefix); ok && existing != "" {
return fmt.Errorf("cannot mount under existing mount '%s'", existing)
}
// If this is a secret backend, check to see if the prefix conflicts
// with an existing mountpoint
if prefix != "" {
Copy link
Member

Choose a reason for hiding this comment

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

I'm pretty sure that we don't allow empty prefixes in the API calls or core mount function. Have you seen this? This if seems unnecessary.

Copy link
Contributor

Choose a reason for hiding this comment

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

Yes, this check appears unnecessary.

vault/router.go Outdated

// MountConflict determines if there are potential path conflicts
func (r *Router) MountConflict(path string) string {
if exact_match := r.MatchingMount(path); exact_match != "" {
Copy link
Member

Choose a reason for hiding this comment

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

Really this function should grab a single read lock for the entire function and then call matchingMountInternal and matchingPrefixInternal. That way nothing can change between the two functions.

@jefferai
Copy link
Member

Regarding the if -- both, actually. I left a comment on the one at line 67. The other one was about the use of HasPrefix in the walking function. Wouldn't we never even hit that part of the walk function if HasPrefix wasn't true?

@jefferai jefferai modified the milestones: 0.7.4, 0.8.0 Jul 24, 2017
@otakup0pe
Copy link
Contributor Author

otakup0pe commented Jul 27, 2017

Shuffled around the "internal" functions. When I removed the prefix matching code in Mount, the build fails and I see a bunch of warnings/errors about audit logs (not sure why I thought policies). They might be a red herring, there is a lot of output to parse when the vault test suite fails.

2017/07/26 22:54:14.121014 [ERROR] audit: backend failed to log request: backend=foo error=failed
2017/07/26 22:54:14.121028 [ERROR] audit: backend failed to log request: backend=foo error=failed
2017/07/26 22:54:14.121037 [ERROR] audit: backend failed to log request: backend=bar error=failed
2017/07/26 22:54:14.121166 [ERROR] audit: backend failed to log response: backend=foo error=failed
2017/07/26 22:54:14.121178 [ERROR] audit: backend failed to log response: backend=foo error=failed
2017/07/26 22:54:14.121185 [ERROR] audit: backend failed to log response: backend=bar error=failed
2017/07/26 22:54:14.121376 [ERROR] audit: backend failed to log request: backend=foo error=failed
2017/07/26 22:54:14.121403 [ERROR] audit: backend failed to log request: backend=foo error=failed
2017/07/26 22:54:14.121430 [ERROR] audit: backend failed to log request: backend=bar error=failed
2017/07/26 22:54:14.078248 [ERROR] core: failed to create audit entry: path=noop/ error=failing enabling
2017/07/26 22:54:14.078259 [ERROR] core: failed to create audit entry: path=noop/ error=failing enabling
2017/07/26 22:54:14.078268 [ERROR] core: failed to create audit entry: path=noop2/ error=failing enabling

@jefferai
Copy link
Member

@otakup0pe Your build works fine -- can you explain what you mean about "when I removed the prefix matching code in Mount"?

@jefferai jefferai modified the milestones: next-release, 0.8.2 Aug 18, 2017
@jefferai
Copy link
Member

@otakup0pe Just wanted to check on on this and see if you were going to address the remaining review items.

@jefferai jefferai modified the milestones: 0.8.2, 0.8.3 Aug 31, 2017
@otakup0pe
Copy link
Contributor Author

Hi @jefferai I was off grid for several weeks. Will pick wrap this up now that I'm back!

Jonathan Freedman and others added 3 commits September 17, 2017 12:35
Changed the tests around to better capture this now does. Existing paths
paths are still supported from the perspective of the router, but not
from the perspective of new mounts.
@otakup0pe
Copy link
Contributor Author

I think this addresses the feedback. I adjusted the tests to account for the new pattern.

@jefferai jefferai modified the milestones: 0.8.3, 0.8.4 Sep 25, 2017
@jefferai jefferai requested a review from vishalnayak October 19, 2017 16:56
t.Fatalf("bad: prod/aws/")
}

err = r.Mount(n, "prod/", subMountEntry, view)
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we have a comment here as to why this should pass?

@@ -118,6 +118,11 @@ func TestRouter_Mount(t *testing.T) {
t.Fatalf("err: %v", err)
}

meUUID, err = uuid.GenerateUUID()
Copy link
Contributor

Choose a reason for hiding this comment

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

Can we move this to be placed just above the definition of subMountEntry?

vault/router.go Outdated
@@ -201,14 +200,46 @@ func (r *Router) MatchingMountByAccessor(mountAccessor string) *MountEntry {
// MatchingMount returns the mount prefix that would be used for a path
func (r *Router) MatchingMount(path string) string {
r.l.RLock()
mount, _, ok := r.root.LongestPrefix(path)
var mount = r.matchingMountInternal(path)
Copy link
Contributor

Choose a reason for hiding this comment

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

[Nitpick][Optional] We could do a defer r.l.RUnlock() and return r.matchingMountInternal(path) and avoid the local var mount.

vishalnayak
vishalnayak previously approved these changes Oct 23, 2017
Copy link
Contributor

@vishalnayak vishalnayak left a comment

Choose a reason for hiding this comment

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

Very minor comments. This is looking good to me.

@jefferai jefferai merged commit a40f8c4 into hashicorp:master Nov 6, 2017
chrishoffman pushed a commit that referenced this pull request Nov 7, 2017
* oss/master: (30 commits)
  Handle 'not supplied' case for field type TypeNameString (#3546)
  Fix deprecated cassandra backend tests (#3543)
  changelog++
  auth/aws: Make disallow_reauthentication and allow_instance_migration mutually exclusive (#3291)
  changelog++
  More Mount Conflict Detection (#2919)
  Fix swallowed errors in TestRollbackManager_Join() (#3327)
  changelog++
  added AWS enpoint handling (#3416)
  Seal wrap all root tokens and their leases (#3540)
  Return group memberships of entity during read (#3526)
  Add note on support for using rec keys on /sys/rekey (#3517)
  Add third party tools list to website (#3488)
  Minor client refactoring (#3539)
  changelog++
  Add PKCS8 marshaling to PKI (#3518)
  Update SSH list roles docs (#3536)
  Update gocql dep
  changelog++
  Return role info for each role on pathRoleList (#3532)
  ...
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Mount conflict inconsistency
4 participants