Skip to content

Conversation

@lony2003
Copy link

@lony2003 lony2003 commented Mar 4, 2025

Description

A new library that implement a backend of Spring Cloud Config.

Originally from https://github.com/fangkehou-team/dapr-spring, this library created a backend of SpringCloudConfig just like SpringCloudVault.
The original library only uses secret store as config store api is not stable at that time.
As the configuration api is stable now, the config loader using that api has been implemented, but subscription of configuration has not implemented, just for reservation.

Issue reference

this PR will close: #1225

Checklist

Please make sure you've completed the relevant tasks for this PR, out of the following list:

  • Code compiles correctly
  • Created/updated tests
  • Extended the documentation

@lony2003 lony2003 requested review from a team as code owners March 4, 2025 02:32
@lony2003 lony2003 force-pushed the development-cloudconfig branch 3 times, most recently from 25b3e33 to 2b4022b Compare March 4, 2025 03:14
@salaboy
Copy link
Collaborator

salaboy commented Mar 4, 2025

@lony2003 first of all, thanks so so much for contributing back with this. This is highly appreciated.. I will start adding comments in the PR with small details that we need to fix and questions about how this would work.

It will be great if you can add as part of this PR an example on the spring-boot-examples/ directory showing how this functionality would work. We probably also need tests inside sdk-tests to demonstrate that this works from a client point of view. You will notice that we are heavily relying on Testcontainers for all of this, so it will be quite cool to see if the testcontainers integration is missing something to support this functionality.

import io.dapr.spring.boot.autoconfigure.client.DaprClientProperties;
import io.dapr.spring.boot.autoconfigure.client.DaprConnectionDetails;

class CloudConfigPropertiesDaprConnectionDetails implements DaprConnectionDetails {
Copy link
Collaborator

Choose a reason for hiding this comment

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

@lony2003 do we need to duplicate the DaprConnectionDetails, why are these different for the normal DaprConnectionDetails?

Copy link
Author

Choose a reason for hiding this comment

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

No difference, just because the PropertyDaprConnectionDetails is protected and can't be used in another package.

Copy link
Collaborator

Choose a reason for hiding this comment

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

@artur-ciocanu should we change the PropertyDaprConnectionDetails visibility so it can be reused?

Copy link
Contributor

Choose a reason for hiding this comment

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

@lony2003 there is a pending PR #1341 I hope to get this in so you could use it. Once the PR is merged, I don't think we need this class.

@lony2003
Copy link
Author

lony2003 commented Mar 4, 2025

@lony2003 first of all, thanks so so much for contributing back with this. This is highly appreciated.. I will start adding comments in the PR with small details that we need to fix and questions about how this would work.

It will be great if you can add as part of this PR an example on the spring-boot-examples/ directory showing how this functionality would work. We probably also need tests inside sdk-tests to demonstrate that this works from a client point of view. You will notice that we are heavily relying on Testcontainers for all of this, so it will be quite cool to see if the testcontainers integration is missing something to support this functionality.

Thanks. the code is not completed currently, I created this pull request is for the problems in issue #1225 , I think we need a schema for cloud config url, and have some other things to be discussed😊😊😊

@lony2003
Copy link
Author

lony2003 commented Mar 5, 2025

Cloud Config Import Schemas

Secret Store Component

url structure

dapr:secret:<store-name>[/<secret-name>][?<paramters>]

paramters

parameter description default available
type value type value value/doc
doc-type type of doc properties yaml/properties/or any file extensions you want
  • when type = value, if secret-name is specified, will treat secret as the value of property, and secret-name as the key of property; if none secret-name is specified, will get bulk secret and treat every value of secret as the value of property, and every key of secret as the key of property.
  • when type = doc, if secret-name is specified, will treat secret as a bunch of property, and load it with property or yaml loader; if none secret-name is specified, will get bulk secret and and treat every value of secret as bunches of property, and load them with property or yaml loader.
  • secret store with multiValud = true must specify nestedSeparator = ".", and using type = doc is not recommanded

demo

multiValued = false:

store content(file secret store as example)
{
	"dapr.spring.demo-config-secret.singlevalue": "testvalue",
	"multivalue-properties": "dapr.spring.demo-config-secret.multivalue.v1=spring\ndapr.spring.demo-config-secret.multivalue.v2=dapr",
	"multivalue-yaml": "dapr:\n  spring:\n    demo-config-secret:\n      multivalue:\n        v3: cloud"
}
valid demo url
  • dapr:secret:democonfig/multivalue-properties?type=doc&doc-type=properties
  • dapr:secret:democonfig/multivalue-yaml?type=doc&doc-type=yaml
  • dapr:secret:democonfig/dapr.spring.demo-config.singlevalue?type=value
  • dapr:secret:democonfig?type=value
  • dapr:secret:democonfig?type=doc

multiValued = true, nestedSeparator = ".":

store content(file secret store as example)
{
	"value1": {
		"dapr": {
			"spring": {
				"demo-config-secret": {
					"multivalue": {
						"v4": "config"
					}
				}
			}
		}
	}
}

will be read as

{
	"value1": {
		"dapr.spring.demo-config-secret.multivalue.v4": "config"
	}
}
valid demo url
  • dapr:secret:demo-config-multi/value1?type=value
  • dapr:secret:demo-config-multi?type=value

Configuration Component

url structure

dapr:config:<store-name>[/<key>][?<paramters>]

paramters

parameter description default available
type value type value doc/value
doc-type type of doc properties yaml/properties/or any file extensions you want
subscribe subscribe this configuration false true/false
  • when subscribe = true, will subscribe update for the configuration.
  • when type = value, if key is specified, will treat config value as the value of property, and key as the key of property; if none key is specified, will get all key and value in the config-name and treat every config value as the value of property, and every key as the key of property.
  • when type = doc, if key is specified, will treat config value as a bunch of property, and load it with property or yaml loader; if none key is specified, will get all key and value in the config-name and treat every config value as bunches of property, and load them with property or yaml loader.

Demo

store content(table as example)

key value
dapr.spring.demo-config-config.singlevalue testvalue
multivalue-properties dapr.spring.demo-config-config.multivalue.v1=spring\ndapr.spring.demo-config-config.multivalue.v2=dapr
multivalue-yaml dapr:\n spring:\n demo-config-config:\n multivalue:\n v3: cloud

valid demo url

  • dapr:config:democonfigconf/dapr.spring.demo-config-config.singlevalue?type=value
  • dapr:config:democonfigconf/multivalue-properties?type=doc&doc-type=properties
  • dapr:config:democonfigconf/multivalue-yaml?type=doc&doc-type=yaml
  • dapr:config:democonfigconf?type=doc
  • dapr:config:democonfigconf?type=value

@cicoyle cicoyle added this to the v1.15 milestone Mar 6, 2025
@lony2003 lony2003 force-pushed the development-cloudconfig branch 7 times, most recently from 383a076 to 034baad Compare March 8, 2025 16:57
@lony2003
Copy link
Author

lony2003 commented Mar 9, 2025

All jobs have done, waiting for review. Note that subscription of configuration api is not implemented, just as a reservation.

@lony2003 lony2003 changed the title [WIP] Spring Boot Cloud Config [All jobs done] Spring Boot Cloud Config Mar 9, 2025
@lony2003
Copy link
Author

lony2003 commented Mar 27, 2025

I have checked the error, and it seems that jacoco coverage throws the error. I have add a test of bulk secret to coverage more situation in DaprSecretStoreConfigDataLoader.java listed in the first line of codeconv report. Not sure if it's the cause of jacoco coverage failed, but I can't add more coverage as it's a little difficult to test error situation in bootstrap stage (springboot application will crash and test will be failed)

@lony2003 lony2003 requested a review from salaboy March 28, 2025 10:43
@salaboy
Copy link
Collaborator

salaboy commented Mar 29, 2025

@lony2003 this is amazing! thank so much for all the work that you put on this PR!
I really appreciate the demo application in the spring boot examples, but I have one more ask. Can you make sure that this also works on Kubernetes?
I want to make sure that people can understand how this will work on Kubernetes when compared with the local version.. Following something like this: https://github.com/dapr/java-sdk/tree/master/spring-boot-examples/kubernetes
what are the steps that people will need to perform on a cluster to configure Spring Cloud Config inside Kubernetes, assuming that the config parameters are in Kubernetes ConfigMaps or Secrets. Does this make sense?

lony2003 and others added 10 commits April 3, 2025 18:01
add a test of Dapr Secret getBulkSecret method

Signed-off-by: lony2003 <[email protected]>
Fix the example doc to make cloud config demo works both local and kubernetes, and pass the mm doc validate

Signed-off-by: lony2003 <[email protected]>
Fix the example doc to make cloud config demo works both local and kubernetes

Signed-off-by: lony2003 <[email protected]>
@lony2003 lony2003 force-pushed the development-cloudconfig branch from 2bed2a2 to 70d4eea Compare April 3, 2025 10:06
salaboy
salaboy previously approved these changes May 7, 2025
Copy link
Collaborator

@salaboy salaboy left a comment

Choose a reason for hiding this comment

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

LGTM

@codecov
Copy link

codecov bot commented May 7, 2025

Codecov Report

Attention: Patch coverage is 73.65591% with 98 lines in your changes missing coverage. Please review.

Project coverage is 75.31%. Comparing base (d759c53) to head (70d4eea).
Report is 138 commits behind head on master.

Files with missing lines Patch % Lines
...igdata/secret/DaprSecretStoreConfigDataLoader.java 68.25% 15 Missing and 5 partials ⚠️
...g/DaprConfigurationConfigDataLocationResolver.java 70.68% 10 Missing and 7 partials ⚠️
...data/config/DaprConfigurationConfigDataLoader.java 68.08% 10 Missing and 5 partials ⚠️
...oudconfig/config/DaprCloudConfigClientManager.java 48.00% 7 Missing and 6 partials ⚠️
...ret/DaprSecretStoreConfigDataLocationResolver.java 85.71% 5 Missing and 3 partials ⚠️
...onfig/configdata/DaprCloudConfigParserHandler.java 85.00% 3 Missing and 3 partials ⚠️
...ta/config/DaprConfigurationConfigDataResource.java 60.00% 6 Missing ⚠️
...data/secret/DaprSecretStoreConfigDataResource.java 57.14% 6 Missing ⚠️
.../cloudconfig/config/DaprCloudConfigProperties.java 76.47% 4 Missing ⚠️
...t/examples/cloudconfig/CloudConfigApplication.java 33.33% 2 Missing ⚠️
... and 1 more
Additional details and impacted files
@@             Coverage Diff              @@
##             master    #1230      +/-   ##
============================================
- Coverage     76.91%   75.31%   -1.60%     
- Complexity     1592     1707     +115     
============================================
  Files           145      211      +66     
  Lines          4843     5473     +630     
  Branches        562      594      +32     
============================================
+ Hits           3725     4122     +397     
- Misses          821     1004     +183     
- Partials        297      347      +50     

☔ View full report in Codecov by Sentry.
📢 Have feedback on the report? Share it here.

🚀 New features to boost your workflow:
  • ❄️ Test Analytics: Detect flaky tests, report on failures, and find test suite problems.

@salaboy
Copy link
Collaborator

salaboy commented May 11, 2025

@dapr/maintainers-java-sdk I believe that we can merge this PR, we will incrementally improve the quality of this PR, but I want this module to be present in the SDK. Please approve, we can fix any findings in smaller PRs.

Copy link
Contributor

@artur-ciocanu artur-ciocanu left a comment

Choose a reason for hiding this comment

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

@lony2003 thanks a lot for your contribution! I have reviewed the PR and I have left quite a few comments.

The main issue that I want to highlight is that we need to make sure that we can leverage ConfigurableBootstrapContext and BootstrapRegistryInitializer to ensure we create the DaprClient and DaprClientProperties beans. As far as I saw a lot of complexity is related to the fact that we try to jump through the hoops just to get the right instances so we could load the data.

While I have mentioned DaprClient and DaprClientProperties beans, all the other dependencies like parse handler should be registered with the ConfigurableBootstrapContext as well, so we could look them up when needed.

Using the aforementioned approach will allow us to simplify the code and be able to expand in the future without a lot of effort.


import io.dapr.spring.data.DaprKeyValueAdapterResolver;
import org.springframework.boot.context.properties.ConfigurationProperties;

Copy link
Contributor

Choose a reason for hiding this comment

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

I would prefer using a simple string prefix like "dapr.client"

<description>Dapr Spring Cloud Config</description>
<packaging>jar</packaging>

<dependencies>
Copy link
Contributor

Choose a reason for hiding this comment

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

As far as I can tell ONLY dapr-spring-boot-autoconfigure dependency should be required all the others seems to be copy-pasted from other POM.

Could we please make sure that ONLY the required dependencies are used.

import io.dapr.spring.boot.autoconfigure.client.DaprClientProperties;
import io.dapr.spring.boot.autoconfigure.client.DaprConnectionDetails;

class CloudConfigPropertiesDaprConnectionDetails implements DaprConnectionDetails {
Copy link
Contributor

Choose a reason for hiding this comment

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

@lony2003 there is a pending PR #1341 I hope to get this in so you could use it. Once the PR is merged, I don't think we need this class.

@@ -0,0 +1,9 @@
package io.dapr.springboot.examples.cloudconfig;

public class DaprConfigurationStores {
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 this extracted to src/test/resources

@@ -0,0 +1,23 @@
package io.dapr.springboot.examples.cloudconfig;

public class DaprSecretStores {
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 extract this to src/test/resources

" \"multivalue-properties\": \"dapr.spring.demo-config-secret.multivalue.v1=spring\\ndapr.spring.demo-config-secret.multivalue.v2=dapr\",\n" +
" \"multivalue-yaml\": \"dapr:\\n spring:\\n demo-config-secret:\\n multivalue:\\n v3: cloud\"\n" +
"}";

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 extract this to src/test/resources


@SpringBootApplication
public class TestCloudConfigApplication {

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 main() method?

name: dapr.spring.demo-config-secret.singlevalue
type: Opaque
stringData:
dapr.spring.demo-config-secret.singlevalue: "testvalue" No newline at end of file
Copy link
Contributor

Choose a reason for hiding this comment

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

Could you please add new line

@artur-ciocanu
Copy link
Contributor

@lony2003 also could you please sync your branch withmaster branch, your branch is out of date.

@lony2003
Copy link
Author

@artur-ciocanu I am working on my graduation currently, once I have time, I‘ll resolve them.

@salaboy Thank you for approving this.

@salaboy
Copy link
Collaborator

salaboy commented May 13, 2025

@lony2003 as soon as you have the changes done please let me know

@salaboy
Copy link
Collaborator

salaboy commented May 13, 2025

@artur-ciocanu thank so much for all the reviews and unblocking

@cicoyle
Copy link
Contributor

cicoyle commented Jun 11, 2025

@lony2003 - can you take a look at CI?

@lony2003
Copy link
Author

@cicoyle I'm working on this, I just merged the base and haven't changed my code. I'll notify you once I have finished

@siri-varma siri-varma modified the milestones: v1.15, v1.16 Aug 20, 2025
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.

A Cloud Config Client for SpringBoot, like Spring Cloud Vault

7 participants