Client

The Dapr client package allows you to interact with other Dapr applications from a .NET application.

Note

If you haven’t already, try out one of the quickstarts for a quick walk-through on how to use the Dapr .NET SDK with an API building block.

Building blocks

The .NET SDK allows you to interface with all of the Dapr building blocks.

Invoke a service

HTTP

You can either use the DaprClient or System.Net.Http.HttpClient to invoke your services.

Note

You can also invoke a non-Dapr endpoint using either a named HTTPEndpoint or an FQDN URL to the non-Dapr environment.

  1. using var client = new DaprClientBuilder().
  2. UseTimeout(TimeSpan.FromSeconds(2)). // Optionally, set a timeout
  3. Build();
  4. // Invokes a POST method named "deposit" that takes input of type "Transaction"
  5. var data = new { id = "17", amount = 99m };
  6. var account = await client.InvokeMethodAsync<object, Account>("routing", "deposit", data, cancellationToken);
  7. Console.WriteLine("Returned: id:{0} | Balance:{1}", account.Id, account.Balance);
  1. var client = DaprClient.CreateInvokeHttpClient(appId: "routing");
  2. // To set a timeout on the HTTP client:
  3. client.Timeout = TimeSpan.FromSeconds(2);
  4. var deposit = new Transaction { Id = "17", Amount = 99m };
  5. var response = await client.PostAsJsonAsync("/deposit", deposit, cancellationToken);
  6. var account = await response.Content.ReadFromJsonAsync<Account>(cancellationToken: cancellationToken);
  7. Console.WriteLine("Returned: id:{0} | Balance:{1}", account.Id, account.Balance);

gRPC

You can use the DaprClient to invoke your services over gRPC.

  1. using var cts = new CancellationTokenSource(TimeSpan.FromSeconds(20));
  2. var invoker = DaprClient.CreateInvocationInvoker(appId: myAppId, daprEndpoint: serviceEndpoint);
  3. var client = new MyService.MyServiceClient(invoker);
  4. var options = new CallOptions(cancellationToken: cts.Token, deadline: DateTime.UtcNow.AddSeconds(1));
  5. await client.MyMethodAsync(new Empty(), options);
  6. Assert.Equal(StatusCode.DeadlineExceeded, ex.StatusCode);

Save & get application state

  1. var client = new DaprClientBuilder().Build();
  2. var state = new Widget() { Size = "small", Color = "yellow", };
  3. await client.SaveStateAsync(storeName, stateKeyName, state, cancellationToken: cancellationToken);
  4. Console.WriteLine("Saved State!");
  5. state = await client.GetStateAsync<Widget>(storeName, stateKeyName, cancellationToken: cancellationToken);
  6. Console.WriteLine($"Got State: {state.Size} {state.Color}");
  7. await client.DeleteStateAsync(storeName, stateKeyName, cancellationToken: cancellationToken);
  8. Console.WriteLine("Deleted State!");

Query State (Alpha)

  1. var query = "{" +
  2. "\"filter\": {" +
  3. "\"EQ\": { \"value.Id\": \"1\" }" +
  4. "}," +
  5. "\"sort\": [" +
  6. "{" +
  7. "\"key\": \"value.Balance\"," +
  8. "\"order\": \"DESC\"" +
  9. "}" +
  10. "]" +
  11. "}";
  12. var client = new DaprClientBuilder().Build();
  13. var queryResponse = await client.QueryStateAsync<Account>("querystore", query, cancellationToken: cancellationToken);
  14. Console.WriteLine($"Got {queryResponse.Results.Count}");
  15. foreach (var account in queryResponse.Results)
  16. {
  17. Console.WriteLine($"Account: {account.Data.Id} has {account.Data.Balance}");
  18. }

Publish messages

  1. var client = new DaprClientBuilder().Build();
  2. var eventData = new { Id = "17", Amount = 10m, };
  3. await client.PublishEventAsync(pubsubName, "deposit", eventData, cancellationToken);
  4. Console.WriteLine("Published deposit event!");

Interact with output bindings

  1. using var client = new DaprClientBuilder().Build();
  2. // Example payload for the Twilio SendGrid binding
  3. var email = new
  4. {
  5. metadata = new
  6. {
  7. emailTo = "customer@example.com",
  8. subject = "An email from Dapr SendGrid binding",
  9. },
  10. data = "<h1>Testing Dapr Bindings</h1>This is a test.<br>Bye!",
  11. };
  12. await client.InvokeBindingAsync("send-email", "create", email);

Retrieve secrets

  1. var client = new DaprClientBuilder().Build();
  2. // Retrieve a key-value-pair-based secret - returns a Dictionary<string, string>
  3. var secrets = await client.GetSecretAsync("mysecretstore", "key-value-pair-secret");
  4. Console.WriteLine($"Got secret keys: {string.Join(", ", secrets.Keys)}");
  1. var client = new DaprClientBuilder().Build();
  2. // Retrieve a key-value-pair-based secret - returns a Dictionary<string, string>
  3. var secrets = await client.GetSecretAsync("mysecretstore", "key-value-pair-secret");
  4. Console.WriteLine($"Got secret keys: {string.Join(", ", secrets.Keys)}");
  5. // Retrieve a single-valued secret - returns a Dictionary<string, string>
  6. // containing a single value with the secret name as the key
  7. var data = await client.GetSecretAsync("mysecretstore", "single-value-secret");
  8. var value = data["single-value-secret"]
  9. Console.WriteLine("Got a secret value, I'm not going to be print it, it's a secret!");

Get Configuration Keys

  1. var client = new DaprClientBuilder().Build();
  2. // Retrieve a specific set of keys.
  3. var specificItems = await client.GetConfiguration("configstore", new List<string>() { "key1", "key2" });
  4. Console.WriteLine($"Here are my values:\n{specificItems[0].Key} -> {specificItems[0].Value}\n{specificItems[1].Key} -> {specificItems[1].Value}");
  5. // Retrieve all configuration items by providing an empty list.
  6. var specificItems = await client.GetConfiguration("configstore", new List<string>());
  7. Console.WriteLine($"I got {configItems.Count} entires!");
  8. foreach (var item in configItems)
  9. {
  10. Console.WriteLine($"{item.Key} -> {item.Value}")
  11. }

Subscribe to Configuration Keys

  1. var client = new DaprClientBuilder().Build();
  2. // The Subscribe Configuration API returns a wrapper around an IAsyncEnumerable<IEnumerable<ConfigurationItem>>.
  3. // Iterate through it by accessing its Source in a foreach loop. The loop will end when the stream is severed
  4. // or if the cancellation token is cancelled.
  5. var subscribeConfigurationResponse = await daprClient.SubscribeConfiguration(store, keys, metadata, cts.Token);
  6. await foreach (var items in subscribeConfigurationResponse.Source.WithCancellation(cts.Token))
  7. {
  8. foreach (var item in items)
  9. {
  10. Console.WriteLine($"{item.Key} -> {item.Value}")
  11. }
  12. }

Distributed lock (Alpha)

Acquire a lock

  1. using System;
  2. using Dapr.Client;
  3. namespace LockService
  4. {
  5. class Program
  6. {
  7. [Obsolete("Distributed Lock API is in Alpha, this can be removed once it is stable.")]
  8. static async Task Main(string[] args)
  9. {
  10. var daprLockName = "lockstore";
  11. var fileName = "my_file_name";
  12. var client = new DaprClientBuilder().Build();
  13. // Locking with this approach will also unlock it automatically, as this is a disposable object
  14. await using (var fileLock = await client.Lock(DAPR_LOCK_NAME, fileName, "random_id_abc123", 60))
  15. {
  16. if (fileLock.Success)
  17. {
  18. Console.WriteLine("Success");
  19. }
  20. else
  21. {
  22. Console.WriteLine($"Failed to lock {fileName}.");
  23. }
  24. }
  25. }
  26. }
  27. }

Unlock an existing lock

  1. using System;
  2. using Dapr.Client;
  3. namespace LockService
  4. {
  5. class Program
  6. {
  7. static async Task Main(string[] args)
  8. {
  9. var daprLockName = "lockstore";
  10. var client = new DaprClientBuilder().Build();
  11. var response = await client.Unlock(DAPR_LOCK_NAME, "my_file_name", "random_id_abc123"));
  12. Console.WriteLine(response.status);
  13. }
  14. }
  15. }

Manage workflow instances (Alpha)

  1. var daprClient = new DaprClientBuilder().Build();
  2. string instanceId = "MyWorkflowInstance1";
  3. string workflowComponentName = "dapr"; // alternatively, this could be the name of a workflow component defined in yaml
  4. string workflowName = "MyWorkflowDefinition";
  5. var input = new { name = "Billy", age = 30 }; // Any JSON-serializable value is OK
  6. // Start workflow
  7. var startResponse = await daprClient.StartWorkflowAsync(instanceId, workflowComponentName, workflowName, input);
  8. // Terminate workflow
  9. await daprClient.TerminateWorkflowAsync(instanceId, workflowComponentName);
  10. // Get workflow metadata
  11. var getResponse = await daprClient.GetWorkflowAsync(instanceId, workflowComponentName, workflowName);

Sidecar APIs

Sidecar Health

The .NET SDK provides a way to poll for the sidecar health, as well as a convenience method to wait for the sidecar to be ready.

Poll for health

This health endpoint returns true when both the sidecar and your application are up (fully initialized).

  1. var client = new DaprClientBuilder().Build();
  2. var isDaprReady = await client.CheckHealthAsync();
  3. if (isDaprReady)
  4. {
  5. // Execute Dapr dependent code.
  6. }

Poll for health (outbound)

This health endpoint returns true when Dapr has initialized all its components, but may not have finished setting up a communication channel with your application.

This is best used when you want to utilize a Dapr component in your startup path, for instance, loading secrets from a secretstore.

  1. var client = new DaprClientBuilder().Build();
  2. var isDaprComponentsReady = await client.CheckOutboundHealthAsync();
  3. if (isDaprComponentsReady)
  4. {
  5. // Execute Dapr component dependent code.
  6. }

Wait for sidecar

The DaprClient also provides a helper method to wait for the sidecar to become healthy (components only). When using this method, it is recommended to include a CancellationToken to allow for the request to timeout. Below is an example of how this is used in the DaprSecretStoreConfigurationProvider.

  1. // Wait for the Dapr sidecar to report healthy before attempting use Dapr components.
  2. using (var tokenSource = new CancellationTokenSource(sidecarWaitTimeout))
  3. {
  4. await client.WaitForSidecarAsync(tokenSource.Token);
  5. }
  6. // Perform Dapr component operations here i.e. fetching secrets.

Shutdown the sidecar

  1. var client = new DaprClientBuilder().Build();
  2. await client.ShutdownSidecarAsync();