aws-secrets-manager-credentials-provider-plugin
aws-secrets-manager-credentials-provider-plugin copied to clipboard
secret names with slashes are not viewable from credentials manager (404 error)
Jenkins and plugins versions report
Environment
Jenkins: 2.303.1
OS: Linux - 5.10.104-linuxkit
---
ace-editor:1.1
apache-httpcomponents-client-4-api:4.5.13-1.0
aws-credentials:191.vcb_f183ce58b_9
aws-java-sdk:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-cloudformation:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-codebuild:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-ec2:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-ecr:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-ecs:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-elasticbeanstalk:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-iam:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-logs:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-minimal:1.12.201-326.veb_6ce41104a_e
aws-java-sdk-ssm:1.12.201-326.veb_6ce41104a_e
aws-secrets-manager-credentials-provider:1.0.0
bootstrap4-api:4.6.0-5
bootstrap5-api:5.1.3-6
bouncycastle-api:2.20
branch-api:2.1046.v0ca_37783ecc5
caffeine-api:2.9.3-65.v6a_47d0f4d1fe
checks-api:1.7.4
cloudbees-folder:6.714.v79e858ef76a_2
command-launcher:1.2
config-file-provider:3.8.1
configuration-as-code:1.55
credentials:2.6.1.1
credentials-binding:1.27.1
display-url-api:2.3.5
durable-task:496.va67c6f9eefa7
echarts-api:5.3.2-1
font-awesome-api:6.0.0-1
git:4.4.1
git-client:3.11.0
git-server:1.11
golang:1.4
jackson2-api:2.13.2.20220328-281.v9ecc7a_5e834f
javax-activation-api:1.2.0-3
javax-mail-api:1.6.2-6
jaxb:2.3.6-1
jdk-tool:1.0
jquery3-api:3.6.0-3
jsch:0.1.55.2
junit:1.48
mailer:414.vcc4c33714601
matrix-project:1.18
nodejs:1.4.0
pipeline-aws:1.43
pipeline-graph-analysis:1.11
pipeline-input-step:448.v37cea_9a_10a_70
pipeline-model-api:1.9.3
pipeline-model-definition:1.7.2
pipeline-model-extensions:1.9.3
pipeline-rest-api:2.24
pipeline-stage-step:293.v200037eefcd5
pipeline-stage-tags-metadata:1.9.3
pipeline-utility-steps:2.6.1
plain-credentials:1.8
plugin-util-api:2.16.0
popper-api:1.16.1-3
popper2-api:2.11.5-1
prisma-cloud-jenkins-plugin:21.08.525
scm-api:608.vfa_f971c5a_a_e9
script-security:1138.v8e727069a_025
slack:2.47
snakeyaml-api:1.30.1
sonar:2.14
splunk-devops:1.9.4
splunk-devops-extend:1.9.4
ssh-agent:1.23
ssh-credentials:277.v95c2fec1c047
sshd:3.228.v4c9f9e652c86
structs:318.va_f3ccb_729b_71
timestamper:1.13
token-macro:293.v283932a_0a_b_49
trilead-api:1.0.13
variant:1.4
workflow-api:1144.v61c3180fa_03f
workflow-basic-steps:2.24
workflow-cps:2660.vb_c0412dc4e6d
workflow-cps-global-lib:2.21.3
workflow-durable-task-step:2.40
workflow-job:1145.v7f2433caa07f
workflow-multibranch:712.vc169a_1387405
workflow-scm-step:400.v6b_89a_1317c9a_
workflow-step-api:625.vd896b_f445a_f8
What Operating System are you using (both controller, and any agents involved in the problem)?
amazon linux 2
Reproduction steps
In an AWS secrets manager account that your Jenkins instance on AWS can access, create a secret with a name like my/secret/text
and give it the correct tags to be read into Jenkins.
Expected Results
- Going to
https://<your-jenkins>/credentials
and clicking on the secret should show it in the normal managed secrets view.
Actual Results
404 error
Anything else?
This appears to be a fairly basic URI-encoding problem, because if you replace the slashes in the URL with %2F
the secret management URL is visible.
I've written a test case for this here: https://github.com/jenkinsci/aws-secrets-manager-credentials-provider-plugin/pull/206
Unfortunately I am unable to reproduce the error, as the test has worked just fine. Would you be able to take the test case from that PR and tweak it, to see if you can reproduce it?
I have the same issue as reported here on my Jenkins instance too.
The credentials work as in builds are able to use them, and they show up when visiting the credentials view https://<your-jenkins>/credentials
, e.g.
... but you can't click on one of those credentials and see the details in Jenkins. e.g. clicking on "Secret for github" says:
However, if I hand-edit the URL to replace the /
in the credential name with %2F
then it works:
I've taken a look at the unit test added in #206 and it seems that it merely checks that a credential is usable (it shows up in the lists that builds can choose from), not that the credential can be inspected using the Jenkins credentials admin Web UI.
I think that what's needed is a unit-test that has a secret called foo/bar with description "FooBar" and verifies that the https://<your-jenkins>/credentials
URL returns a page containing a link which has text FooBar
pointing to a URL ending /foo%2Fbar/
instead of /foo/bar/
The Secrets Manager provider isn't the only plugin in the chain, so I'm just wondering if maybe the URL encoding handling is going wrong upstream...
Could you try creating a local credential in the default (on disk) credentials provider with the name that you used before (/i2eng/cicd...
), then try browsing to it in the same way, and let me know if that fails due to the same error?
I think this says it all...
If you use the built-in credentials provider, it tries to stop you you creating credentials with a /
in the ID.
The AWS credentials provider doesn't stop you and, in many ways, folks are encouraged to use this kind of path with AWS secrets, thus causing this issue...
...however, if I ignore the red warning and create it anyway (with username "foo", password "bar" and description "Foo/bar") then it gets created successfully:
...and the link on that page to edit the credential works OK - it has all the
/
characters replaced with %2F
in the URL already.
The fact that it worked despite the red warning suggests that the red warning was a false-alarm (so maybe there was a time when it wasn't a false-alarm and there's been code added to support /
in the path since?)
HOWEVER, if I then browse back to just /credentials/
:
...then the link there does NOT work - that page doesn't have the
/
replaced with %2F
and clicking it causes a 404.
i.e. the basic credential provider partly works with IDs containing a /
- when listing credentials for the built-in provider's global credential then the links work, but when listing "all credentials in Jenkins" then the links don't.
...but if I inspect the AWS provider's global credentials then the links don't work there (or on the "all credentials in Jenkins" page).
TL;DR: The links are 50% borked with the built-in Jenkins credential provider but 100% borked with the AWS provider. i.e. it's a bit of a mess...
Yes, we'll need to go digging in the default credential provider code to see if it has special handling for /
characters. Not only to see what they did, but also to reuse it if we can - it's important to be consistent with that handling code to make future maintenance easier.
This is the bit of credentials-plugin that causes the validation warning you saw:
public final FormValidation doCheckId(@ContextInPath ModelObject context, @QueryParameter String value) {
if (value.isEmpty()) {
return FormValidation.ok();
}
if (!value.matches("[a-zA-Z0-9_.-]+")) {
return FormValidation.error("Unacceptable characters");
}
// ....
https://github.com/jenkinsci/credentials-plugin/blob/589d6391ed041908a49023033d7982f71dbc1042/src/main/java/com/cloudbees/plugins/credentials/impl/BaseStandardCredentials.java#L201
It's not targeting the '/' character in particular, and there's nothing in the associated git commit to suggest it was. Rather, it's a general limitation to ASCII alphanumeric plus a couple of delimiters they decided to allow (underscores, periods, or hyphens).
Yup, and there's even an old JIRA report complaining that it's demanding "a too narrow set of characters", https://issues.jenkins.io/browse/JENKINS-45254 ...but that JIRA notes "Sadly the source code does not indicate any reasons why this regex was added in the first place." 😞
My guess is that there is a bug in the credential plugin where it fails to correctly encode the URL in one of the two places where credentials are listed ... but it gets it right in the other place, and we need to figure out how that works so that the AWS plugin can copy that trick. (and if that then reveals how/why it's not working with the top-level view then that's help get that one fixed in the credentials plugin too ... or maybe the answer lies 100% within the credentials plugin?)
If I had to guess, I imagine it was added as a basic 'sanitise your inputs' (OWASP bread and butter secure coding) measure. In that school of thought, the fewer characters you allow the better, and I guess few people were using '/' as a namespacer back then.
The hunt continues for the url encoding bit in credentials-plugin... (alternatively it could be upstream in credentials-api)
The problem is that it doesn't "sanitise [your] inputs" because it doesn't actually prevent creation of credentials with illegal IDs. Sure, it displays a red error complaining about it, but the "submit" button works and creates the credential with the nasty ID. i.e. it adds inconvenience and worries the user, but doesn't provide any security by policing the IDs. Plus, of course, the basic credential provider can't stop other credential providers (like this AWS one) from creating IDs that the basic provider wouldn't like.
(credentials-api? There's another layer too? Well, I guess it's worth a look, given that I got nowhere fast with the credentials plugin itself...)
Yes, credentials-api is the plugin that contains the credentials interfaces, which credentials consumers interact with.
credentials-plugin, aws-secrets-manager-credentials-provider-plugin and so forth contain the concrete providers which implement those interfaces.
It's possible that URL encoding is being handled by a proxy class or method in credentials-api
The nested class CredentialsStoreAction$CredentialsWrapper
in credentials-plugin
(that's the concrete implementation) looks promising...
It overrides the describable's method public String getUrlName()
, and performs some kind of encoding on the credential ID.
If indeed this is it, it explains how it could simultaneously urlencode the credential ID for the Web UI, and make the unmodified credential ID available to direct consumers (because if everything received a urlencoded ID, they would probably break).
Edit: Link is https://github.com/jenkinsci/credentials-plugin/blob/b03fbc98f341564f8e663c01184a9f9a8e3891a5/src/main/java/com/cloudbees/plugins/credentials/CredentialsStoreAction.java#L1150
/**
* A wrapper object to bind and expose {@link Credentials} instances into the web UI.
*/
@ExportedBean
public static class CredentialsWrapper extends AbstractDescribableImpl<CredentialsWrapper>
implements IconSpec, ModelObjectWithContextMenu, AccessControlled {
/**
* Our {@link DomainWrapper}.
*/
private final DomainWrapper domain;
/**
* The {@link Credentials} that we are wrapping.
*/
private final Credentials credentials;
/**
* The {@link IdCredentials#getId()} of the {@link Credentials}.
*/
private final String id;
private Fingerprint fingerprint;
/**
* Constructor.
*
* @param domain the wrapped domain.
* @param credentials the credentials.
* @param id the id.
*/
public CredentialsWrapper(DomainWrapper domain, Credentials credentials, String id) {
this.domain = domain;
this.credentials = credentials;
this.id = id;
}
/**
* Return the id for the XML API.
*
* @return the id.
* @since 2.1.0
*/
@Exported
public String getId() {
return id;
}
/**
* Return the URL name.
*
* @return the URL name.
*/
public String getUrlName() {
return Util.rawEncode(id);
}
...
That seems a plausible explanation 👍
...but only for one of the two cases - it doesn't explain why /
characters can mess things up for one of the two credential-listings - I suspect one of them is calling getId()
and the other is calling getUrlName()
...
I have asked the author of Azure Keyvault Plugin whether that credentials provider supports IDs with /
characters, and if so how they are urlencoded.
https://github.com/jenkinsci/azure-keyvault-plugin/issues/115
It turns out that Azure Keyvault Plugin conveniently sidesteps this problem, because in Azure Keyvault secrets are not allowed to have /
in their IDs anyway.
This is because the URL structure of Keyvault works like S3; there's one global namespace:
https://{vault-name}.vault.azure.net/{secret-type}/{secret-name}/{secret-version}
And vault-name
is typically your product name, with an environment suffix e.g. myproduct-staging
, byproduct-production
.
Started a PR to fix this: #218
At the moment that PR is very much non-functional while we try to figure out how the CredentialsWrapper is used in the bigger picture of the CredentialsStore.
I've also opened https://issues.jenkins.io/browse/JENKINS-68817 to ask if it would be appropriate to move the CredentialsWrapper logic up into the Credentials API, so that urlencoding is done in a uniform way regardless of which provider is used.
Hi @Peter-Darton-i2
I circled back to this today to see if anything had changed in more recent Jenkins versions, and Jenkins Credentials (API) versions.
I tried to create a local secret in the default credentials provider (so not the AWS one) called foo/bar
. Like before, it rejected this because '/' is an invalid character.
I went ahead and created it with a different name, then shut Jenkins down, edited the credential's name in credentials.xml
to be foo/bar
, and started Jenkins again.
You can see that foo/bar
is visible in the list:
data:image/s3,"s3://crabby-images/fb791/fb791af24101681bbe8f4b9c4cefeae36eeb7632" alt="credentials"
HOWEVER, when you click on foo/bar
to browse to it, you now get a 404 Not Found (the same error that you saw with credentials from Secrets Manager):
data:image/s3,"s3://crabby-images/b6f53/b6f533f88987d092bed38e7bf8d3f17e1dccd99d" alt="system-credential-not-found"
And changing the URL to escape the forward slash (http://localhost:8080/.../credential/foo%2Fbar/
) makes it work again - just like you saw with the Secrets Manager plugin before.
This was seen with:
- Jenkins 2.361.1
- Credentials Plugin 1189.vf61b_a_5e2f62e
I think that if the default credentials provider is doing this, then this is now the standard expected behaviour across all providers. It's not something that can be handled within the Secrets Manager plugin, it would need to be dealt with upstream.
Fortunately this is only a minor issue for the Secrets Manager plugin, because the detailed credential view is mostly used to reach its editing form. In the Secrets Manager plugin credentials are read-only, so they are never edited within Jenkins anyway. It therefore doesn't block any critical paths of operation. (But this is a bigger issue for the default credentials provider which does rely on the editing form.)
I will therefore close this issue, but before I do, I think it's worth mentioning something about this in the README as a known limitation.
I have added this clarification to the README in https://github.com/jenkinsci/aws-secrets-manager-credentials-provider-plugin/pull/259
Although it's not exactly the fix you guys were hoping for (because that can only be implemented upstream), I hope that it will at least prevent confusion for future users of this plugin who encounter this behaviour. If you have any input on the documentation change PR, it would be appreciated.
The documentation clarification is now merged - closing this PR