providers/claude/plugin/skills/apideck-dotnet/SKILL.md
Apideck Unified API integration patterns for C# and .NET. Use when building integrations with accounting software (QuickBooks, Xero, NetSuite), CRMs (Salesforce, HubSpot, Pipedrive), HRIS platforms (Workday, BambooHR), file storage (Google Drive, Dropbox, Box), ATS systems (Greenhouse, Lever), e-commerce, or any of Apideck's 200+ connectors using .NET. Covers the ApideckUnifySdk NuGet package, authentication, CRUD operations, pagination, error handling, and Vault connection management.
npx skillsauth add apideck-libraries/api-skills apideck-dotnetInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
The Apideck Unified API provides a single integration layer to connect with 200+ third-party services across accounting, CRM, HRIS, file storage, ATS, e-commerce, and more. The official .NET SDK (ApideckUnifySdk) provides typed clients for all unified APIs.
dotnet add package ApideckUnifySdk
ApideckUnifySdk NuGet package. DO NOT make raw HttpClient calls to the Apideck API.apiKey, appId, and consumerId when initializing the client.ServiceId to specify which downstream connector to use (e.g., "salesforce", "quickbooks").Task<T>.BaseException as the base class.using ApideckUnifySdk;
using ApideckUnifySdk.Models.Requests;
var sdk = new Apideck(
consumerId: "your-consumer-id",
appId: "your-app-id",
apiKey: Environment.GetEnvironmentVariable("APIDECK_API_KEY") ?? ""
);
var res = await sdk.Crm.Contacts.ListAsync(new CrmContactsAllRequest {
ServiceId = "salesforce",
Limit = 20,
});
while (res != null)
{
foreach (var contact in res.GetContactsResponse?.Data ?? [])
{
Console.WriteLine($"{contact.Name} - {contact.Emails?.FirstOrDefault()?.Email}");
}
res = await res.Next!();
}
using ApideckUnifySdk;
var sdk = new Apideck(
consumerId: "your-consumer-id",
appId: "your-app-id",
apiKey: Environment.GetEnvironmentVariable("APIDECK_API_KEY") ?? ""
);
All resources follow the pattern: sdk.{Api}.{Resource}.{Operation}Async(request).
using ApideckUnifySdk;
using ApideckUnifySdk.Models.Requests;
using ApideckUnifySdk.Models.Components;
// LIST
var listRes = await sdk.Crm.Contacts.ListAsync(new CrmContactsAllRequest {
ServiceId = "salesforce",
Limit = 20,
Filter = new ContactsFilter { Email = "[email protected]" },
Sort = new ContactsSort {
By = ContactsSortBy.UpdatedAt,
Direction = SortDirection.Desc,
},
});
// CREATE
var createRes = await sdk.Crm.Contacts.CreateAsync(new CrmContactsAddRequest {
ServiceId = "salesforce",
Contact = new ContactInput {
FirstName = "John",
LastName = "Doe",
Emails = new List<Email> {
new Email { EmailAddress = "[email protected]", Type = EmailType.Primary },
},
PhoneNumbers = new List<PhoneNumber> {
new PhoneNumber { Number = "+1234567890", Type = PhoneNumberType.Mobile },
},
},
});
Console.WriteLine(createRes.CreateContactResponse?.Data?.Id);
// GET
var getRes = await sdk.Crm.Contacts.GetAsync(new CrmContactsOneRequest {
Id = "contact_123",
ServiceId = "salesforce",
});
// UPDATE
var updateRes = await sdk.Crm.Contacts.UpdateAsync(new CrmContactsUpdateRequest {
Id = "contact_123",
ServiceId = "salesforce",
Contact = new ContactInput { FirstName = "Jane" },
});
// DELETE
await sdk.Crm.Contacts.DeleteAsync(new CrmContactsDeleteRequest {
Id = "contact_123",
ServiceId = "salesforce",
});
Use the Next method on the response. Returns null when no more pages:
var res = await sdk.Accounting.Invoices.ListAsync(new AccountingInvoicesAllRequest {
ServiceId = "quickbooks",
Limit = 50,
});
while (res != null)
{
foreach (var invoice in res.GetInvoicesResponse?.Data ?? [])
{
Console.WriteLine($"{invoice.Number}: {invoice.Total}");
}
res = await res.Next!();
}
using ApideckUnifySdk.Models.Errors;
try
{
var res = await sdk.Crm.Contacts.GetAsync(new CrmContactsOneRequest { Id = "invalid" });
}
catch (BadRequestResponse e)
{
Console.Error.WriteLine($"Bad request: {e.Message}");
}
catch (UnauthorizedResponse e)
{
Console.Error.WriteLine("Invalid API key or missing credentials");
}
catch (NotFoundResponse e)
{
Console.Error.WriteLine("Record not found");
}
catch (PaymentRequiredResponse e)
{
Console.Error.WriteLine("API limit reached");
}
catch (UnprocessableResponse e)
{
Console.Error.WriteLine($"Validation error: {e.Message}");
}
catch (BaseException e)
{
Console.Error.WriteLine($"API error: {e.Message}");
Console.Error.WriteLine($"Status: {e.Response.StatusCode}");
}
var sdk = new Apideck(
retryConfig: new RetryConfig(
strategy: RetryConfig.RetryStrategy.BACKOFF,
backoff: new BackoffStrategy(
initialIntervalMs: 1L,
maxIntervalMs: 50L,
maxElapsedTimeMs: 100L,
exponent: 1.1
),
retryConnectionErrors: false
),
consumerId: "your-consumer-id",
appId: "your-app-id",
apiKey: Environment.GetEnvironmentVariable("APIDECK_API_KEY") ?? ""
);
| Namespace | Resources |
|-----------|-----------|
| sdk.Accounting.* | Invoices, Bills, Payments, Customers, Suppliers, LedgerAccounts, JournalEntries, TaxRates, CreditNotes, PurchaseOrders, BalanceSheet, ProfitAndLoss, and more |
| sdk.Crm.* | Contacts, Companies, Leads, Opportunities, Activities, Notes, Pipelines, Users |
| sdk.Hris.* | Employees, Companies, Departments, Payrolls, TimeOffRequests |
| sdk.FileStorage.* | Files, Folders, Drives, DriveGroups, SharedLinks, UploadSessions |
| sdk.Ats.* | Applicants, Applications, Jobs |
| sdk.Vault.* | Connections, Consumers, Sessions, CustomMappings, Logs |
| sdk.Webhook.* | Webhooks, EventLogs |
development
Jira Teams via Apideck's Proxy API + managed Vault auth — Apideck handles auth and proxies HTTP calls to Jira Teams's native API. Use when the user wants to call Jira Teams (no unified API resource mapping). Routes through Apideck with serviceId "jira-teams".
development
Jira Service Desk via Apideck's Proxy API + managed Vault auth — Apideck handles auth and proxies HTTP calls to Jira Service Desk's native API. Use when the user wants to call Jira Service Desk (no unified API resource mapping). Routes through Apideck with serviceId "jira-service-desk".
development
Jira Data Center via Apideck's Proxy API + managed Vault auth — Apideck handles auth and proxies HTTP calls to Jira Data Center's native API. Use when the user wants to call Jira Data Center (no unified API resource mapping). Routes through Apideck with serviceId "jira-data-center".
development
JetBrains YouTrack via Apideck's Proxy API + managed Vault auth — Apideck handles auth and proxies HTTP calls to JetBrains YouTrack's native API. Use when the user wants to call JetBrains YouTrack (no unified API resource mapping). Routes through Apideck with serviceId "jetbrains-youtrack".