mirror of
https://github.com/github/awesome-copilot.git
synced 2026-05-15 11:11:48 +00:00
6.1 KiB
6.1 KiB
Microsoft Graph SDK for .NET
Use this reference when the target project is written in C# or another .NET language.
Authoritative sources
- SDK repository: https://github.com/microsoftgraph/msgraph-sdk-dotnet
- Samples: https://github.com/microsoftgraph/msgraph-training-dotnet
- SDK changelog: https://github.com/microsoftgraph/msgraph-sdk-dotnet/blob/main/CHANGELOG.md
Packages
<!-- Microsoft Graph SDK v5 (current) -->
<PackageReference Include="Microsoft.Graph" Version="5.*" />
<!-- Azure Identity for credential providers -->
<PackageReference Include="Azure.Identity" Version="1.*" />
Install via CLI:
dotnet add package Microsoft.Graph
dotnet add package Azure.Identity
Client setup
Managed Identity (Azure-hosted apps — preferred)
using Azure.Identity;
using Microsoft.Graph;
var credential = new DefaultAzureCredential();
var graphClient = new GraphServiceClient(credential);
Client credentials (app-only / daemon)
var credential = new ClientSecretCredential(
tenantId: Environment.GetEnvironmentVariable("AZURE_TENANT_ID"),
clientId: Environment.GetEnvironmentVariable("AZURE_CLIENT_ID"),
clientSecret: Environment.GetEnvironmentVariable("AZURE_CLIENT_SECRET")
);
var graphClient = new GraphServiceClient(credential);
Prefer ClientCertificateCredential over ClientSecretCredential in production.
On-Behalf-Of (OBO) — agent / API acting as the signed-in user
// incomingToken is the bearer token received from the caller
var credential = new OnBehalfOfCredential(
tenantId: Environment.GetEnvironmentVariable("AZURE_TENANT_ID"),
clientId: Environment.GetEnvironmentVariable("AZURE_CLIENT_ID"),
clientSecret: Environment.GetEnvironmentVariable("AZURE_CLIENT_SECRET"),
userAssertion: new UserAssertion(incomingToken)
);
var graphClient = new GraphServiceClient(credential);
Interactive (local dev / CLI)
var credential = new InteractiveBrowserCredential();
var graphClient = new GraphServiceClient(credential);
Common call patterns
Get a resource with field selection
var user = await graphClient.Me.GetAsync(config =>
{
config.QueryParameters.Select = ["displayName", "mail", "jobTitle"];
});
List with filter and select
var messages = await graphClient.Me.Messages.GetAsync(config =>
{
config.QueryParameters.Filter = "isRead eq false";
config.QueryParameters.Select = ["subject", "from", "receivedDateTime"];
config.QueryParameters.Top = 25;
config.QueryParameters.Orderby = ["receivedDateTime desc"];
});
Pagination with PageIterator
var messages = await graphClient.Me.Messages.GetAsync();
var allMessages = new List<Message>();
var pageIterator = PageIterator<Message, MessageCollectionResponse>
.CreatePageIterator(graphClient, messages, (msg) =>
{
allMessages.Add(msg);
return true; // return false to stop early
});
await pageIterator.IterateAsync();
Send an email
await graphClient.Me.SendMail.PostAsync(new SendMailPostRequestBody
{
Message = new Message
{
Subject = "Hello from Graph",
Body = new ItemBody { ContentType = BodyType.Text, Content = "Test message" },
ToRecipients = [new Recipient { EmailAddress = new EmailAddress { Address = "user@contoso.com" } }]
}
});
Post a Teams channel message
await graphClient.Teams[teamId].Channels[channelId].Messages.PostAsync(new ChatMessage
{
Body = new ItemBody { ContentType = BodyType.Html, Content = "<b>Hello from Graph!</b>" }
});
Batch requests
using Microsoft.Graph.Models;
var batchRequestContent = new BatchRequestContentCollection(graphClient);
var meRequest = await batchRequestContent.AddBatchRequestStepAsync(
graphClient.Me.ToGetRequestInformation());
var messagesRequest = await batchRequestContent.AddBatchRequestStepAsync(
graphClient.Me.Messages.ToGetRequestInformation());
var batchResponse = await graphClient.Batch.PostAsync(batchRequestContent);
var me = await batchResponse.GetResponseByIdAsync<User>(meRequest);
var msgs = await batchResponse.GetResponseByIdAsync<MessageCollectionResponse>(messagesRequest);
Delta queries
// First sync — get all + deltaLink
var deltaResponse = await graphClient.Users.Delta.GetAsDeltaGetResponseAsync();
string? deltaLink = null;
var pageIterator = PageIterator<User, Microsoft.Graph.Users.Delta.DeltaGetResponse>
.CreatePageIterator(graphClient, deltaResponse, (user) => { /* process */ return true; },
(req) => { deltaLink = /* extract from response */; return req; });
await pageIterator.IterateAsync();
// Store deltaLink for next run
// Subsequent sync — only changes
// Use the stored deltaLink directly as the next request URL
Throttling / retry middleware
The SDK includes retry middleware enabled by default. For explicit control:
var handlers = GraphClientFactory.CreateDefaultHandlers();
// RetryHandler is included; configure max retries if needed
var httpClient = GraphClientFactory.Create(handlers);
var graphClient = new GraphServiceClient(httpClient, credential);
Always check Retry-After if building custom retry logic — do not use fixed exponential backoff.
Dependency injection (ASP.NET Core / .NET Worker)
// Program.cs
builder.Services.AddSingleton<GraphServiceClient>(_ =>
{
var credential = new DefaultAzureCredential();
return new GraphServiceClient(credential);
});
.NET-specific guidance
- Target .NET 8+ for new projects.
- Use
async/awaitthroughout — all Graph SDK calls are async. - Register
GraphServiceClientas a singleton (it caches tokens internally). - Use
ILoggerto log Graph exceptions — catchODataErrorfor Graph-specific error details. - For ASP.NET Core APIs using OBO, inject the incoming token from
IHttpContextAccessorand construct the credential per-request (not as a singleton).
// Catching Graph errors
try
{
var user = await graphClient.Me.GetAsync();
}
catch (ODataError odataError)
{
Console.WriteLine($"Graph error: {odataError.Error?.Code} - {odataError.Error?.Message}");
}