-
-
Notifications
You must be signed in to change notification settings - Fork 4.1k
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
Add cloud id resolver interface #35732
Conversation
The obvious question to start is why are |
$posSlash = strpos($id, '/'); | ||
$posColon = strpos($id, ':'); | ||
|
||
if ($posSlash === false && $posColon === false) { | ||
$invalidPos = \strlen($id); | ||
} elseif ($posSlash === false) { | ||
$invalidPos = $posColon; | ||
} elseif ($posColon === false) { | ||
$invalidPos = $posSlash; | ||
} else { | ||
$invalidPos = min($posSlash, $posColon); | ||
} | ||
|
||
$lastValidAtPos = strrpos($id, '@', $invalidPos - strlen($id)); |
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.
It’s not even clear to me what this code is supposed to do. The strpos dance is quite complex.
@schiessle Do you remember why these characters were forbidden?
@PVince81 @blizzz @icewind1991 Do you think allowing these characters in here can lead to trouble? They are allowed in group names so it does make sense.
In which case I would vote for removing this whole block as invalid userids should be catched elsewhere and here we should just split on the @
sign.
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.
Puh, that's really a good question.
But whatever you do, keep in mind that you can't just split at the first @-sign. People can use email addresses as uids, so a valid federated cloud ID could look like user@mailserver@nextcloudserver
. Maybe it would be safe to always split on the last @-sign but maybe I miss some other edge cases?
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.
Yeah that’s what the strrpos is doing, finding the last @
sign.
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.
Yes, the question for me is: why don't we just search for the last @-sign but have this offset? Do we break something when we remove the offset? My gut feeling says that it should be fine. But I hardly believe that we come up with all this logic to calculate the offset without a good reason. That's why I still fear that we miss something.
Can we find the pull request which introduced this code? Maybe the description or discussion gives us a idea why it was done this way?
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 just want to give a short recount what $lastValidAtPos = strrpos($id, '@', $invalidPos - strlen($id))
does.
- If there is no slash or colon,
strrpos
just returns the last position of at sign (or false). - Otherwise the search window for at sign is reduced by cutting off the part after the first occurrence of colon or slash (negative offset).
So user@https://example.com
and user@example.com@https://example.com
are both valid while urn:isbn:0451450523@https://example.com
is not.
Maybe the reason for having this complex logic was to support URLs with userinfo like user@https://username:password@example.com
. This would correctly identify the first at sign as lastValidAtPos while a simple strrpos($cloudId, '@')
does not.
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.
Can we find the pull request which introduced this code? Maybe the description or discussion gives us a idea why it was done this way?
This one might be relevant to the discussion: 2b7e5f8
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.
Instead of validating all possible cloud ids there is now a mechanism to register cloud id resolvers.
Validation has become more user-centric in 256fbe9.
Does anyone know if there is a documented limit of size for federated cloud id? |
@come-nc there is likely a limit, either 512 or 255 chars, potentially shorter, see oc_share_external column "remote" and "remote_id" |
a283542
to
2fe2427
Compare
oc_share_external.user does have a limitation of 64 characters. There is a migration in #34132 to increase it to 255 characters in order to support long group names. |
/** | ||
* Check if the input is a correctly formatted cloud id | ||
* | ||
* @param string $cloudId | ||
* @return bool | ||
* | ||
* @since 26.0.0 | ||
*/ | ||
public function isValidCloudId(string $cloudId): bool; |
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.
If the resolver returns true it indicates to the CloudIdManager that this cloud id should also be resolved by it. This is a cursory check and an exception can still be thrown during resolveCloudId
Signed-off-by: Sandro Mesterheide <sandro.mesterheide@extern.publicplan.de>
2fe2427
to
6deb049
Compare
* | ||
* @since 26.0.0 | ||
*/ | ||
public function registerCloudIdResolver(ICloudIdResolver $resolver); |
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.
public function registerCloudIdResolver(ICloudIdResolver $resolver); | |
public function registerCloudIdResolver(ICloudIdResolver $resolver): void; |
* | ||
* @since 26.0.0 | ||
*/ | ||
public function unregisterCloudIdResolver(ICloudIdResolver $resolver); |
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.
public function unregisterCloudIdResolver(ICloudIdResolver $resolver); | |
public function unregisterCloudIdResolver(ICloudIdResolver $resolver): void; |
namespace OC\Federation; | ||
|
||
use OCP\Federation\ICloudId; | ||
namespace OCP\Federation; | ||
|
||
class CloudId implements ICloudId { |
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.
class CloudId implements ICloudId { | |
/** | |
* @since 27.0.0 | |
*/ | |
class CloudId implements ICloudId { |
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.
/** | ||
* @param ICloudIdResolver $resolver | ||
* | ||
* @since 26.0.0 |
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.
* @since 26.0.0 | |
* @since 27.0.0 |
/** | ||
* @param ICloudIdResolver $resolver | ||
* | ||
* @since 26.0.0 |
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.
* @since 26.0.0 | |
* @since 27.0.0 |
/** | ||
* @param ICloudIdResolver $resolver | ||
*/ | ||
public function unregisterCloudIdResolver(ICloudIdResolver $resolver) { |
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.
public function unregisterCloudIdResolver(ICloudIdResolver $resolver) { | |
public function unregisterCloudIdResolver(ICloudIdResolver $resolver): void { |
/** | ||
* @param ICloudIdResolver $resolver | ||
*/ | ||
public function registerCloudIdResolver(ICloudIdResolver $resolver) { |
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.
public function registerCloudIdResolver(ICloudIdResolver $resolver) { | |
public function registerCloudIdResolver(ICloudIdResolver $resolver): void { |
@@ -100,6 +104,12 @@ public function handleCardEvent(Event $event): void { | |||
*/ | |||
public function resolveCloudId(string $cloudId): ICloudId { | |||
// TODO magic here to get the url and user instead of just splitting on @ | |||
|
|||
foreach ($this->cloudIdResolvers as $resolver) { |
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.
Not sure if there is a better way, but isValidCloudId will be called twice on each resolver, since it is then called by $this->isValidCloudId
. Not ideal performance-wise long-term.
Hello @smesterheide , looks like this PR was marked as "stale". But would like to reach out since there was already quite some work put here and would encourage you to ping me here to see if you'd like to reopen! 🚀 |
Summary
This is a companion/sub PR of #34132. An application requires group ids that contain slashes and colons. Full reasoning given:
Changes
TODO
Checklist