/reference-architecture-secure-cross-tenant-API-traffic

A reference architecture for secure cross-tenant API traffic, preserves the idea of private network connections, and makes use of a single API Management instance to centralize and manage all API traffic for an organization.

OtherNOASSERTION

Reference architecture for secure cross-tenant API traffic

CC BY-NC-SA 4.0 GitHub commit activity

The Azure landscape steadily moves to Microsoft managed product-suites. Previously, we had the option to provision Azure Purview in our own Azure subscription, but now we have Microsoft Purview. Similarly, Azure Synapse Analytics and Data Factory were available for provisioning in our subscription, but now we have Microsoft Fabric. This shift isn't merely about rebranding; it represents a move from managing Azure resources within our own subscription to managing them within Microsoft's tenant. As a consequence, there's an implicit cross-tenancy relationship with Microsoft across various levels, including the network.

To maintain the concept of traditional private network connections in this cross-tenant environment, Microsoft introduced the concept of managed private endpoints. These endpoints allow secure connections from Microsoft's network, via Managed Virtual Networks in their tenant, to our own Virtual Networks.

With the increasing standardization of traffic flowing through networks via API protocols, APIs and their traffic are becoming the focal point of modern IT architecture. Well-designed APIs and their traffic are now regarded as first-class citizens in mature IT architectures, given the interconnected nature of the modern world through APIs.

In this landscape, centralized tooling becomes crucial to simplify API traffic and securely connect APIs across tenants. Azure Virtual Network remains a powerful tool for facilitating network connections to the cloud or on-premises, while Azure API Management serves as an excellent platform for centralizing API traffic.

The following reference architecture for secure cross-tenant API traffic preserves the idea of private network connections, and makes use of a single API Management instance to centralize and manage all API traffic for an organization. This setup not only eliminates the need of IaaS gateway products like self-hosted integration runtimes that facilitate bridged networking, it also tries to set the standard for what a good tenant architecture for APIs looks like.

Technical reference architecture

Following is the technical reference architecture diagram. The diagram consist of the following components:

  • The Microsoft managed resource block signifies the Microsoft tenant, which is managed by Microsoft, but where the customer has some ownership over and can trust as traffic of their own.

  • The public internet and partner managed resources block signifies the traditional network landscape, with no direct link to a tenant, nor to Azure. This is traffic we cannot fully trust.

  • The customer managed resource block signifies the tenant you own, where you have full ownership over. Which contains WAF functionality to filter and control the traditional network landscape outside of your own, and an Azure Application Gateway where traditional and modern cross-tenant traffic comes together.

    • In addition, an Azure API Management resource is placed behind the Azure Application Gateway to centralize standardized APIs. This Azure API management resource differentiates and controls the flow of traffic through product polices.
    • Finally, Microsoft Entra ID is placed as the authorization component for secure APIs for internal use. Microsoft Entra External ID is placed as the authorization component for partner use.

The diagram:

flowchart TD

subgraph Partner managed resources
    netwp["Partner network"]
end

subgraph Public internet
    sbi["Any network"]
end

subgraph Microsoft managed resources
    subgraph Microsoft Fabric
        we["Synapse Pipeline"]
    end
    
    subgraph Managed Virtual Network   
        mpe[Managed Private Endpoint] 
    end

    we --> mpe
end

subgraph Customer managed resources

    netwc["Customer network"]

    netwc --> privatefi

    subgraph Azure Application Gateway
        pubblicfi["Public frontend IP"]
        privatefi["Private frontend IP"]
    end

    waf["Azure Web Application Firewall"]

    mpe -- cross-tenant <br>Private Endpoint Connection --> privatefi
    netwp --> waf 
    sbi --> waf
    waf --> pubblicfi

    subgraph Azure API Management
        subgraph Private product
            apiprivate["TimeRegistration API"]
        end

        subgraph Partner product
            apipartner["Accountant API"]
        end

        subgraph Public product
            apipublic["MyPublicData API"]
        end
    end

    privatefi -- APIM network policy --> apiprivate

    pubblicfi -- APIM network policy --> apipartner
    pubblicfi -- APIM rate policy --> apipublic

    entraid["Microsoft Entra ID"]
    entraidext["Microsoft Entra External ID"]

end

apiprivate -- JWT RBAC --> entraid
apipartner -- JWT RBAC --> entraidext

Loading

Reference implementation for Microsoft managed origins

Following is a reference implementation for traffic originating from Microsoft managed resources, as a part of the architecture.

Prerequisites

Notes

  • The configuration steps make use of HTTP on port 80. The intended implementation is HTTPS on port 443, this works, but make sure there is a globally trusted certificate used so resources connecting to the managed private endpoint understand the CA. A custom certificate might not work.
  • The configuration steps are through click-ops and only intended to be followed like so in a sandbox environment. In production environments, consider using a mature IaC tool like Bicep, Terraform or Bicep# for a streamlined Bicep experience.

Configuration steps

After following the prerequisites, create the connection. This can be done with the following steps:

  1. Create a fully working private path in the Application Gateway to a backend. This is not optional, an empty configuration will not allow you to finish the Private link configuration. This means configuring:

    1. a private fronted IP configuration, in this example it's called appGwPrivateFrontendIpIPv4.
    2. a listener on the private fronted IP configuration, listening on apimanagement.contoso.nl for example.
    3. a backend (pool and settings), pointing towards your backend service, for instance an API Management private IP.
    4. an associated rule to chain together the listener to the backend.
  2. Create a Private link configuration in the Application Gateway. This will enable (managed) private endpoint connection support for the selected Frontend IP configuration, either the Public, or Private type, but it only makes sense to select Private type in order to preserve a secure downstream connection. In the following example, the appGwPrivateFrontendIpIPv4 is configured.

    appgw-privatelinkconfig

  3. Create a managed private endpoint for Application Gateway, in this example we create one in Synapse. The target sub-resource is appGwPrivateFrontendIpIPv4 because we enabled private endpoint connection support for it in the previous step. It should look like this after configuration:

    synapse-mpe-provisioning

  4. Approve the connection in the Application Gateway:

    appgw-mpe-approve

  5. Take notice of the managed private endpoint in Synapse after the approval, there is now a managed private endpoint private IP with value 10.250.0.16, this is our entry point to the Application Gateway and is a private IP residing in the Managed Virtual Network of Synapse:

    synapse-mpe-provisioned

  6. All there is left is using the right settings to utilize the managed private endpoint in Synapse. Because there is no concept of DNS resolution for managed private endpoints for Application Gateway, use the managed private endpoint private IP that was created earlier for the URL field and form a working request, for instance http://10.250.0.16/hello-api/helloworld. The multi-site setup does require a valid host, so the host header is used. This setup assumes:

    • the frontend in the Application Gateway listens on requests with host apimanagement.contoso.nl on port 80 (as configured in step 1.2)
    • the backend in the Application Gateway accepts GET requests on path hello-api/helloworld.

    In Advanced, make sure to select the runtime that is connected to the managed virtual network.

    In a Web activity it might look like this:

    synapse-webactivity

Securing API Management

In a setup of the Application Gateway, where both public and private traffic flows trough, it is possible to distinguish between public and private traffic from a backend (API Management) perspective. Create a rewrite set in the Application Gateway and make sure every rule has a rewrite that sets a specific header, where the value will be an identifier for listeners with a private frontend IP and listeners with a public frontend IP, it might look like this:

appgw-rewrite

Configure a root or product policy with logic to identify private traffic based on the newly introduced header with the Check HTTP header policy in API Management, in a similar way as documented for Front Door.

Considerations

  1. The client_ip or add_x_forwarded_for_proxy server variable in the Application Gateway is appended with 10.250.0.16 in this example. This is not a practical way to check the source (with an API Management ip-filter policy for instance), as the managed virtual network ranges are not properly documented, nor are the ranges configurable during provisioning. As a side note: IP-spoofing exists but is proven to be nearly impossible on TCP (which HTTP uses, which we only allow in this API architecture), so it is not a wrong method to consider.

  2. An API Management policy can also leverage the IPrivateEndpointConnection context information, but that requires an API Management instance with private endpoint. At this time, using a private endpoint setup restricts the use of outbound VNet integration, so in many cases this isn't a practical setup. Besides that, this example does not connect to an API Management private endpoint, but to a private IP that was selected from the injected VNet, although it should work if you use the private endpoint IP in the backend settings of the Application Gateway.

  3. Azure Front Door might be a candidate to replace Azure Application Gateway in the overall architecture, but it isn't supported for Managed private endpoint connections in Microsoft managed origins and does not support API Management instances with internal mode VNets.

  4. Azure Private Link-service might be a candidate to replace managed private endpoint for Application Gateway, but it requires placing an Azure Load Balancer as a front, which in turn requires to have an Azure Virtual Machine Scale Set or Virtual Machine as a backend, which just increases complexity and unnecessary IaaS components.

  5. To facilitate geo-redundancy in this architecture, Azure API Management can be configured multi-regional with load balancing to backends, but this also requires a geo-redundant front on top of, or as a replacement of, Azure Application Gateway to make the entire flow geo-redundant. This either results in:

    • a combination of Azure Front Door and Azure Private Link-service to replace the Application Gateway, but as consideration 4 shows, it requires an Azure Load Balancer and additional IaaS for the managed private endpoint connection route, and as consideration 3 shows, it requires a different type of API Management setup;
    • an Application Gateway in each region, in combination with setting up a Traffic Manager Profile similar to the setup specified by Paco de la Cruz, but with a key difference; pointing the Traffic Manager Profiles to a DNS hostname that resolves to a healthy private or public Application Gateway IP at all times, so for internal traffic to the frontend private IP address of the healthy Application Gateway by private DNS resolution, and for external traffic to the frontend public IP address by public DNS resolution. By doing this, it preserves logic introduced in the Application Gateway. This setup does not solve automatic failover for the (managed) private endpoint connection route, since these do not allow DNS resolution, these services have to implement failover to managed private endpoints IPs on their own way for the time being, which may not be ideal depending on your requirements.
  6. The Microsoft managed resource block in the diagram could very well consist of resources from a partner. Which, in some cases, you want to trust as some similar concept like AD trust relationships. It would be ideal to draw that connection from the Microsoft managed resource block to a partner API, or even a private API, but there is currently no secure way of recognizing the origin of traffic once it hits an Application Gateway frontend IP, so no further trust can be established at this time. The introduction of an Application Gateway server variable that recognizes the Managed Private Endpoint connection (e.g. client_managed_private_endpoint_id) can solve this, but this is up to Microsoft to implement.

License

This work is licensed under a Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International License.

CC BY-NC-SA 4.0