Skip to content

Security overview

Security overview

This document provides full security documentation for the App Store for Intune, including all Azure resources created, permissions granted, and security configurations. It is intended to assist security teams with reviews and compliance requirements.

  1. Azure resources created
  2. Identity and access management
  3. Microsoft Entra ID app registrations
  4. Managed identity configuration
  5. Network security
  6. Data protection
  7. Authentication flow
  8. Authorization model
  9. Audit logging
  10. Security checklist
  11. Package verification

The ARM template deploys the following resources:

Resource TypeName PatternPurpose
App Service Planasp-apprequest-{env}Hosts the web application
App Serviceapprequest-{env}-{unique}ASP.NET Core API + React SPA
Azure Key Vaultkv-appreq-{unique}Secure secret storage
SQL Serversql-apprequest-{env}-{unique}Database server
SQL DatabaseAppRequestPortalApplication data storage
Storage Accountstappreq{unique}Queue + blob storage for packaging
Application Insightsai-apprequest-{env}Application monitoring
Log Analytics Workspacela-apprequest-{env}Centralized logging
Azure Botbot-apprequest-{env}-{unique}Teams proactive messaging (optional)
Bot Teams Channel{bot}/MsTeamsChannelEnables Teams communication (optional)
  • Runtime: .NET 8.0 on Linux
  • TLS Version: Minimum TLS 1.2 enforced
  • HTTPS Only: Enabled (HTTP redirects to HTTPS)
  • FTPS State: Disabled (no FTP access)
  • Client Affinity: Disabled (stateless application)
  • Always On: Enabled (prevents cold starts)
  • TLS Version: Minimum TLS 1.2
  • Firewall: Only Azure services allowed (0.0.0.0-0.0.0.0)
  • Authentication: SQL authentication (username/password)
  • SKU: F0 (Free)
  • Location: Global
  • App Type: SingleTenant
  • Microsoft App ID: Reuses the Backend API Microsoft Entra ID App Registration (AzureAd:ClientId)
  • Messaging Endpoint: https://{app-url}/api/messages
  • Channels: Microsoft Teams only
  • Authentication: Bot Framework handles its own authentication via ConfigurationBotFrameworkAuthentication using the same Entra ID client credentials as the API

Bot Framework Authentication Model:

  • The Azure Bot resource does not use a separate App Registration. It shares the Backend API registration.
  • Bot Framework validates incoming activities from Teams using the AzureAd:ClientId, AzureAd:ClientSecret, and AzureAd:TenantId configuration values
  • Proactive messages are sent via BotAdapter.ContinueConversationAsync or CreateConversationAsync, authenticated with the same credentials
  • No additional Microsoft Graph API permissions are required for Teams bot notifications. Bot Framework handles its own auth channel.

Data stored:

  • BotConversationReferences table stores per-user conversation references for proactive messaging
  • Contains: User ID (Microsoft Entra ID object ID), Conversation ID, Service URL, serialized ConversationReference JSON
  • Conversation references are encrypted/signed by Bot Framework and tied to the bot credentials that created them
  • If bot credentials change, existing conversation references become invalid and must be cleared

The App Service is configured with a System-Assigned Managed Identity. This identity is automatically created and managed by Azure.

"identity": {
"type": "SystemAssigned"
}

Purpose: The Managed Identity allows the application to authenticate to Azure services without storing credentials in code or configuration.

The following permissions are granted to the Managed Identity:

Permission TypeScopePurpose
Website Contributor RoleApp ServiceEnables in-app updates via Azure Resource Manager
Key Vault Access Policy (Get, List Secrets)Key VaultRead application secrets from Key Vault

Why Website Contributor?

The in-app update feature allows administrators to update the application from within the portal. To do this, the application must be able to modify its own WEBSITE_RUN_FROM_PACKAGE app setting. The Website Contributor role grants this permission.

Why Key Vault Access?

All sensitive secrets (Microsoft Entra ID client secret, SQL connection string, storage connection string) are stored in Azure Key Vault. The Managed Identity needs read access to retrieve these secrets at runtime.

Scope limitation: All permissions are scoped to the minimum required resources (not the resource group or subscription), following the principle of least privilege.

{
"type": "Microsoft.Authorization/roleAssignments",
"scope": "[concat('Microsoft.Web/sites/', variables('appName'))]",
"properties": {
"roleDefinitionId": "[subscriptionResourceId('Microsoft.Authorization/roleDefinitions', 'de139f84-1756-47ae-9be6-808fbbe84772')]",
"principalId": "[reference(resourceId('Microsoft.Web/sites', variables('appName')), '2022-09-01', 'full').identity.principalId]",
"principalType": "ServicePrincipal"
}
}

Two Microsoft Entra ID App Registrations are required:

1. Backend API application (confidential client)

Section titled “1. Backend API application (confidential client)”

Application type: Web application / API

Authentication:

  • Client ID (stored in App Settings)
  • Client Secret (stored in App Settings)

API permissions (application permissions):

APIPermissionTypePurpose
Microsoft GraphDeviceManagementApps.Read.AllApplicationRead Intune apps
Microsoft GraphDeviceManagementApps.ReadWrite.AllApplicationCreate/update Intune apps
Microsoft GraphDeviceManagementConfiguration.Read.AllApplicationRead Intune assignment filters (used by ring deployment settings, required for the filter picker)
Microsoft GraphDeviceManagementManagedDevices.Read.AllApplicationCount managed devices for licensing
Microsoft GraphGroup.ReadWrite.AllApplicationCreate and manage Microsoft Entra ID groups
Microsoft GraphUser.Read.AllApplicationRead user profiles, managers, and group memberships
Microsoft GraphDirectory.Read.AllApplicationRead directory data
Microsoft GraphMail.SendApplicationSend email notifications (optional)

Teams bot note: No additional Microsoft Graph permissions are required for Teams bot notifications. Bot Framework handles its own authentication channel separately from Graph. Do not add TeamsActivity.Send or similar permissions; they are not used by this application.

Admin consent: Required for all application permissions.

Token configuration:

  • Access tokens issued for: api://{client-id}
  • Scopes exposed: access_as_user

2. Frontend SPA application (public client)

Section titled “2. Frontend SPA application (public client)”

Application type: Single-page application (SPA)

Authentication:

  • Client ID only (no secret - public client)
  • MSAL.js with PKCE flow

API permissions (delegated permissions):

APIPermissionTypePurpose
Microsoft GraphUser.ReadDelegatedRead signed-in user’s profile and photo
Backend APIaccess_as_userDelegatedCall backend API on behalf of user

Redirect URIs:

  • https://{app-name}.azurewebsites.net/
  • https://{custom-domain}/ (if custom domain configured)

  1. Creation: When the ARM template deploys, Azure automatically creates a service principal in Microsoft Entra ID for the App Service.

  2. Environment variables: Azure injects these environment variables into the application:

    • MSI_ENDPOINT or IDENTITY_ENDPOINT: Token endpoint URL
    • IDENTITY_HEADER: Secret header for authentication
    • WEBSITE_SITE_NAME: App Service name
    • WEBSITE_RESOURCE_GROUP: Resource group name
    • WEBSITE_OWNER_NAME: Subscription ID
  3. Token acquisition: The application uses DefaultAzureCredential from Azure.Identity SDK to acquire tokens:

    var credential = new DefaultAzureCredential();
    var armClient = new ArmClient(credential);
  4. Authorization: The Managed Identity authenticates to Azure Resource Manager and is authorized via the Website Contributor role assignment.

  • Read and update the App Service’s own configuration settings
  • Read the App Service’s properties
  • Restart the App Service
  • Read secrets from the provisioned Key Vault (Get, List permissions only)
  • Access other resources in the subscription
  • Modify other App Services
  • Write, delete, or manage secrets in Key Vault
  • Access SQL Database directly (uses SQL authentication via connection string from Key Vault)
  • Deploy or delete resources

ProtocolPortSourceDestinationPurpose
HTTPS443InternetApp ServiceUser access
HTTPS443App ServiceAzure SQLDatabase connections
HTTPS443App ServiceMicrosoft GraphAPI calls
HTTPS443Bot Framework ServiceApp ServiceTeams bot messages (/api/messages)
HTTPS443App ServiceBot Framework ServiceProactive bot notifications
  • Public Access: Enabled (internet-accessible)
  • TLS 1.2: Minimum version enforced
  • HTTPS Only: HTTP requests redirected to HTTPS
  • IP Restrictions: None by default (can be configured)

The SQL Server is configured with the following security controls:

  • Azure Services: Allowed (firewall rule AllowAllWindowsAzureIps)
    • This allows the App Service to connect to the database
    • Other Azure services in the subscription cannot access unless explicitly allowed
  • Public IP Access: Denied by default
    • No client IP addresses are on the allowlist
    • Database is not accessible from the internet
  • Private Endpoint: Not configured by default (can be added for enhanced security)

To verify SQL firewall rules:

Terminal window
az sql server firewall-rule list --server <server-name> --resource-group <rg-name>

To add a client IP for temporary admin access:

Terminal window
az sql server firewall-rule create --server <server-name> --resource-group <rg-name> \
--name "AdminAccess" --start-ip-address <your-ip> --end-ip-address <your-ip>

Remember to remove temporary rules after use.

We strongly recommend configuring the following Microsoft Entra ID Conditional Access policies for the App Store for Intune. Target these policies at the Backend API and Frontend SPA app registrations.

PolicyDescriptionWhy It Matters
Require MFARequire multifactor authentication for all users accessing the portalPrevents unauthorized access from compromised credentials
Require compliant deviceOnly allow access from Intune-compliant devicesSince this portal manages Intune apps, ensuring the requesting device meets compliance is a natural fit
Require managed deviceOnly allow access from Microsoft Entra ID joined or hybrid joined devicesPrevents access from personal/unmanaged devices
Block risky sign-insBlock sign-ins flagged as medium or high risk by Microsoft Entra ID Identity ProtectionAutomatically blocks access when suspicious activity is detected (requires Microsoft Entra ID P2)
Restrict by locationOnly allow access from trusted named locations (office IPs, VPN ranges)Limits exposure to known network boundaries
Sign-in frequencyRequire re-authentication every 8-12 hoursLimits the window of exposure from a stolen session token
  1. Virtual network integration: Deploy App Service into a VNet
  2. Private endpoints: Use private endpoints for SQL and storage
  3. IP restrictions: Limit access to known IP ranges
  4. Azure Front Door: Add WAF protection

Data TypeStorage LocationEncryption
Application dataAzure SQL DatabaseTDE (Transparent Data Encryption)
User settingsAzure SQL DatabaseTDE
Branding assetsAzure SQL DatabaseTDE
Bot conversation referencesAzure SQL DatabaseTDE
LogsLog AnalyticsAzure-managed encryption
  • All communication uses TLS 1.2+
  • HTTPS enforced at App Service level
  • SQL connections use encrypted connections
DataStorage MethodNotes
Microsoft Entra ID Client SecretAzure Key VaultReferenced via Key Vault reference in App Settings
SQL Connection StringAzure Key VaultReferenced via Key Vault reference in App Settings
Storage Connection StringAzure Key VaultReferenced via Key Vault reference in App Settings
User tokensSession storage (browser)Not persisted server-side
Action tokensSQL DatabaseGUID per request, used for email approve/reject
Bot conversation referencesSQL DatabaseConversation IDs and service URLs for Teams proactive messaging
Audit logsSQL DatabaseRetained indefinitely

All sensitive secrets are stored in Azure Key Vault and accessed via Key Vault references:

AzureAd__ClientSecret = @Microsoft.KeyVault(SecretUri=https://{vault}.vault.azure.net/secrets/AzureAdClientSecret/)
ConnectionStrings__DefaultConnection = @Microsoft.KeyVault(SecretUri=https://{vault}.vault.azure.net/secrets/SqlConnectionString/)
AzureStorage__ConnectionString = @Microsoft.KeyVault(SecretUri=https://{vault}.vault.azure.net/secrets/StorageConnectionString/)

Key Vault security features:

  • Secrets encrypted at rest with Azure-managed keys
  • Soft delete enabled (7-day retention for accidental deletion recovery)
  • Access restricted to the App Service’s Managed Identity only
  • No direct secret access for users or administrators (must use Azure Portal/CLI with appropriate permissions)
  • All secret access is logged in Key Vault diagnostic logs

Microsoft Entra ID client secret rotation:

The Microsoft Entra ID client secret has an expiration date (typically 1-2 years). To rotate:

  1. Create new secret in Azure Portal → Microsoft Entra ID → App Registrations → Your API App → Certificates & secrets
  2. Update Key Vault secret:
    Terminal window
    az keyvault secret set --vault-name <vault-name> --name AzureAdClientSecret --value "<new-secret>"
  3. Restart App Service to pick up the new secret:
    Terminal window
    az webapp restart --name <app-name> --resource-group <rg-name>
  4. Delete old secret from Microsoft Entra ID after confirming the app works

SQL password rotation:

  1. Reset password in Azure Portal → SQL Server → Reset admin password
  2. Update connection string in Key Vault:
    Terminal window
    az keyvault secret set --vault-name <vault-name> --name SqlConnectionString --value "Server=tcp:..."
  3. Restart App Service

Emergency key rotation:

If you suspect a secret has been compromised:

  1. Immediately rotate the affected secret using the steps above
  2. Review Key Vault access logs for unauthorized access
  3. Review App Service logs for unusual activity
  4. Consider rotating all secrets if compromise scope is unknown

SSL certificates:

Certificate TypeManagementRenewal
Azure-managed (default)AutomaticAuto-renewed by Azure
Custom domain with Azure certificateAutomaticAuto-renewed by Azure
Custom uploaded certificateManualUpload new certificate before expiration

Microsoft Entra ID app certificates (optional):

If using certificate authentication instead of client secrets:

  • Monitor expiration dates in Microsoft Entra ID → App Registrations → Certificates & secrets
  • Upload new certificate before expiration
  • Update Key Vault with new certificate thumbprint

Monitoring expiration:

Set up Azure Monitor alerts for:

  • Key Vault secret expiration (90 days before)
  • Microsoft Entra ID client secret expiration
  • SSL certificate expiration (if custom)
Terminal window
# Check Key Vault secret expiration dates
az keyvault secret list --vault-name <vault-name> --query "[].{name:name, expires:attributes.expires}"

┌──────────────┐ ┌──────────────┐ ┌──────────────┐
│ Browser │ │ Entra ID │ │ App Service │
└──────┬───────┘ └──────┬───────┘ └──────┬───────┘
│ │ │
│ 1. Navigate to app │ │
│───────────────────────────────────────────────>│
│ │ │
│ 2. Redirect to sign-in│ │
│<───────────────────────────────────────────────│
│ │ │
│ 3. User authenticates │ │
│───────────────────────>│ │
│ │ │
│ 4. Return tokens │ │
│<───────────────────────│ │
│ │ │
│ 5. API call with token │
│───────────────────────────────────────────────>│
│ │ │
│ │ 6. Validate token │
│ │<───────────────────────│
│ │ │
│ │ 7. Token valid │
│ │───────────────────────>│
│ │ │
│ 8. API response │ │
│<───────────────────────────────────────────────│

The API validates JWT tokens from Microsoft Entra ID:

  1. Issuer: Must match Microsoft Entra ID tenant
  2. Audience: Must match API Client ID
  3. Signature: Validated using Microsoft Entra ID public keys
  4. Expiration: Token must not be expired
  5. Claims: User identity extracted from token

RoleHow AssignedCapabilities
UserAny authenticated userBrowse apps, submit requests, view own requests
ApproverMember of Approver Microsoft Entra ID GroupApprove/reject requests, view pending approvals
AdminMember of Admin Microsoft Entra ID GroupFull access, manage apps, settings, users

Authorization groups are configured in the database (PortalSettings table):

AdminGroupId -- Microsoft Entra ID Group Object ID for admins
ApproverGroupId -- Microsoft Entra ID Group Object ID for approvers
UserAccessGroupId -- (Optional) Restrict access to specific group
// Check if user is admin
var userId = User.FindFirstValue("oid");
var isAdmin = await _azureADService.IsUserInGroupAsync(userId, adminGroupId);

EventData Captured
App request submittedUser ID, App ID, Device ID, Timestamp
Request approved/rejectedReviewer ID, Decision, Comments, Timestamp
Settings changedUser ID, Setting changed, Old/New values
App syncUser ID, Apps added/updated, Timestamp
User sign-inUser ID, IP address, Timestamp
  • Location: SQL Database (AuditLogs table)
  • Retention: Indefinite (no automatic purge)
  • Access: Admin-only via Reports section
CREATE TABLE AuditLogs (
Id NVARCHAR(450) PRIMARY KEY,
UserId NVARCHAR(450) NOT NULL,
UserEmail NVARCHAR(255) NOT NULL,
Action NVARCHAR(100) NOT NULL,
EntityType NVARCHAR(100) NOT NULL,
EntityId NVARCHAR(450) NOT NULL,
Details NVARCHAR(MAX), -- JSON payload
Timestamp DATETIME2 NOT NULL,
IpAddress NVARCHAR(50) NOT NULL
);

  • Microsoft Entra ID App Registrations created with minimum required permissions
  • Admin consent granted for application permissions
  • Strong SQL administrator password configured
  • Admin Group ID configured (required, admin access is denied without it)
  • Approver Group ID configured (restricts approver access)
  • Verify HTTPS-only access working
  • Test authentication flow
  • Verify role-based access (admin vs. user)
  • Review Microsoft Entra ID sign-in logs
  • Enable Azure Security Center recommendations
  • If Teams bot enabled: verify Azure Bot App ID matches AzureAd:ClientId
  • If Teams bot enabled: test bot notification delivery
  • Rotate Microsoft Entra ID client secret annually
  • Review audit logs regularly
  • Monitor failed authentication attempts
  • Update application when security patches released
  • Review and remove unused app permissions
  • Enable Microsoft Entra ID Conditional Access policies (see Recommended Policies)
  • Configure Microsoft Entra ID Identity Protection (required for risk-based CA policies)
  • Implement Private Endpoints for SQL and Key Vault
  • Add Azure Front Door with WAF
  • Enable Azure Defender for App Service
  • Azure Key Vault for secrets (enabled by default)

Apps deployed to Intune via the App Store originate as YAML manifests in the microsoft/winget-pkgs community repository. Each manifest declares an installer URL and the SHA256 hash that installer must match. The packaging pipeline verifies the downloaded installer against that hash before wrapping it for Intune (PackagingQueueService.DownloadFromGitHubAsync).

That hash check defends against a tampered installer at the publisher’s download server, but it trusts the manifest itself. The manifest is fetched from raw.githubusercontent.com over HTTPS, and anyone able to MITM that connection (or substitute a manifest in a compromised mirror) could feed the pipeline a manifest that points at a malicious installer with a matching hash.

To close that gap, the App Store can verify each package against Microsoft’s signed community index. The WinGet client itself uses this same chain (see PackagesTable.h in winget-cli for the upstream schema).

How it works (when VerifyManifestIntegrity is enabled):

  1. The portal downloads source2.msix from https://cdn.winget.microsoft.com/cache/source2.msix.
  2. The .msix contains a SQLite database (Public/index.db) with a packages table: one row per package Microsoft has reviewed and merged into the community repository, with the SHA256 of the package’s version data manifest in the hash column.
  3. Before packaging any WinGet-sourced app, the portal looks the package up in this index. If the package is not present, the packaging job fails with an explicit “Manifest integrity check failed” error rather than silently falling back to an unverified path.
  4. The downloaded MSIX and extracted SQLite are cached on disk for 12 hours so repeat lookups don’t re-download the ~10 MB archive.

What this protects against:

  • Tampered manifest substituted on the GitHub fetch path. A malicious manifest for a non-existent or non-merged package ID will fail the index lookup.
  • An attacker pointing an existing package ID at a different installer. Once paired with the version-data-manifest hash check (planned, see roadmap below), the manifest contents can be cryptographically tied back to what Microsoft moderators reviewed.

What this does not yet protect against (v1 limitations):

  • The full design (documented internally as the WinGet manifest verification specification) calls for fetching the version data manifest from the Azure CDN by content-addressed URL and verifying its hash against the SQLite value, then doing the same for each per-version merged manifest. v1 implements only the index lookup gate; the hash-of-manifest comparison is a planned v2 enhancement.
  • MSIX signature validation against Microsoft’s certificate chain (Authenticode-on-MSIX via WinTrust). v1 relies on HTTPS + DNS to cdn.winget.microsoft.com as the trust root. A compromised CDN is out of scope for v1; v2 will add full signature validation.

Configuration: Admin → Settings → “Verify manifest integrity against Microsoft’s signed index”. Default is off in this release while we soak the feature; once validated in production it will flip to default-on. Admins running internal hardened mirrors of the WinGet repo can leave it off and rely on their mirror’s own integrity controls.

Failure mode: Verification failures are loud, not silent. The packaging job is marked failed and the error message names the package and explains the integrity check failed. There is no fallback to the unverified path while verification is enabled. The only escape valve is to disable the setting from the Settings page.


  • All data stored in the Azure region selected during deployment
  • No cross-region data replication by default
  • Audit logs: Retained indefinitely
  • Application data: Retained until manually deleted
  • App Insights logs: 90 days default (configurable)
  • User data limited to: Name, Email, User ID (from Microsoft Entra ID)
  • No sensitive personal data stored
  • Data can be exported/deleted upon request via database access

Monitor for these events in Microsoft Entra ID and App Insights:

  1. Multiple failed sign-ins from same IP
  2. Unusual geographic access patterns
  3. Bulk data access (many API calls in short time)
  4. Privilege escalation attempts
  5. Configuration changes outside business hours
  1. Disable App Service if compromise suspected
  2. Rotate secrets (Microsoft Entra ID client secret, SQL password)
  3. Review audit logs for unauthorized access
  4. Revoke user access if account compromised
  5. Contact Azure Support for assistance

For security questions or to report vulnerabilities: