This issue was moved to a discussion.
You can continue the conversation there. Go to discussion →
Proposal: Distribute Bicep providers using OCI registry #10662
Labels
design approved
The team has reviewed and signed off on this design
proposal
story: dynamic type loading
Collects all work items related to decoupling of Bicep types from compiler
Milestone
Summary
TL;DR: We propose to change the means of distribution of Bicep provider types from NuGet (static compilation dependency) to OCI registry (dynamic loading).
Motivation
The current implementation of Bicep providers is based on NuGet packages has the following limitations:
Bicep costumers must wait for a Bicep release before they can consume Azure Service providers features made available in new
api-versions
.When a customer upgrades their Bicep compiler to the latest version, previously compiling bicep files may result in compilation failures in the upgraded Bicep compiler because of added linter/compilation checks in the updated version.
As a consequence, Bicep customers that pin their Bicep compiler version to a specific version can't consume new resource
api-versions
and benefit from bug fixes and new features without a significant investment in upgrading their Bicep compiler version.Since the
az
provider definitions are generated, when a newapi-version
is released for an Azure resource, the Bicep team must update theaz
provider definitions and release a new version of theaz
provider NuGet package, if there are defects in the generator or the generated code, the Bicep team must release a new version of Bicep to fix the issue. This makes for a degraded customer experience because of the added time and effort a customer must invest to consume the new Bicep release.As a Bicep user, I want to use the latest
api-versions
of Azure resources without having to upgrade my Bicep compilerAs a Bicep user, I want resource definitions to be immutable so that I control what definitions are used by the Bicep compiler during the compilation process
Design
Current State
Currently,
az
resource type definitions are serialized intotypes.json
in a cron job that's run on a weekly basis.The current implementation based on NuGet packages uses the following workflow:
az
provider types from theazure-rest-api-specs
definitions. The generated artifacts are json blobs conforming to thetypes.json
serialization format, the json files rely on a directory structure and are kept under a folder namedgenerated
(here an there after referred to as the generated-package). The cron job published the updated contents to a branch namedautogenerated
in theazure/bicep-types-az
repository.autogenerated
branch into themain
branch.az
provider NuGet package to NuGet.org from a mirror repository (BicepMirror-Types-Az
) in ADO.az
provider NuGet package.A similar workflow is used the
kubernetes
extensibility provider.Following, we describe how the provider resource type data is loaded into the Bicep compiler.
Azure.Bicep.Types.Az.dll
, this library implements a single classAzTypeLoader
that extends an abstract classTypeLoader
from a different NuGet packageAzure.Bicep.Types
.TypeLoader
is an abstract class that knows how to deserializetypes.json
into runtime objects.Azure.Bicep.Types
NuGet to deserialize the embedded resource into runtime objects that it processes into internal data structures that are used during the compilation process.Proposed State
We propose replacing the NuGet package with an OCI manifest. The OCI registry will contain the same artifacts as the generated-package, the artifacts will be immutable and will be published on a weekly basis by a similar cron job in (
azure/bicep-types-az
). The Bicep compiler will be updated to load the provider resource type data from the OCI registry instead of the NuGet package.The OCI artifact will be a manifest that conforms to the following structure:
The cron job will publish the
az
manifest to a repositorymcr.microsoft.com/bicep/providers/az
hosted by the Microsoft Container Registry (MCR). The Bicep compiler will be updated to load generate the ByteStream from the OCI manifest cache instead of the NuGet package.Implementation Details
The current implementation for Bicep modules loads module data from a filesystem cache under
~/.bicep
. We propose to extend this logic to support loading provider manifests using the same cache. TheCreateCompilation
method of theBicepCompiler
class is the entrypoint for this new logic. The method manipulates asourceFileGrouping
object that traverses the Bicep files that describe the deployment and inspects their syntax for module declarations and loads the data from the filesystem when a module declaration is found. We propose to extend this process to inspect the syntax for provider declarations and restore their data from the filesystem cache.The current implementation for modules uses a class
OciModuleRegistry
to load module data from the network when there is a cache miss. We propose to rename and generalize this class to support loading provider data as well. The strategy used for caching provider data will align with the current implementation for modules, that is, a restore operation will be performed on provider versions that aren't in the cache on demand during thebicep build
processing.az
providerTo maintain backwards compatibility with the current implementation, we propose to ship the Bicep compiler with an embedded version of the provider manifests for
az
andkubernetes
. The embedded providers will be expanded into the filesystem cache and will be used when the user doesn't specify aaz
provider in theimport
directive.Sample expanded folder structure in the local Bicep cache for
az
provider:Scalability or cost concerns with using ACR to store provider data
az
provider generated-package size is ~ 150MB uncompressed, after applying tar+gzip we get a compressed size of ~20MBaz
provider data ARC with geo-replication (see here for costs).Out of scope
- Support for "un-importing" of providers (e.g. removing symbols from the implicitly imported
az
provider)The current proposal is based on the provider model introduced by the extensibility providers feature. It uses the
import
gesture to load symbols of a provider into the current file context. This provider mechanism is used also for Microsoft resources that conform a special case of a provider known as theaz
provider.Encapsulating
az
to its own provider presents its own challenges, in the current implementation of Bicep symbols such as targetScope,
resourceId(),
environment(),
reference()and others are hardcoded into the global scope of Bicep compiler ([see here](https://github.com/Azure/bicep/blob/main/src/Bicep.Core/Semantics/Namespaces/AzNamespaceType.cs#L288-L443)). Other implications of encapsulating these relate to the ability to express them in
types.json(see below bullet), hence, we defer the ability to "un-import" symbols from the
azprovider until a separate work item to encapsulate
az` is implemented.- Serialization of provider scoped functions (e.g.
resourceId()
,environment()
,reference()
) intypes.json
Its concievable for providers to expose provider scoped functions similar to the functions above, however the current serialization format (
types.json
) doesn't support encoding such symbols. The current implementation hardcodesaz
provider function in the Bicep compiler. The current serialization protocol must be enhanced to allow encode provider function signatures before we can dinamically load them from a provider definition. This will be handled in a separate proposal.- Aliasing or configuring aliases for provider registries in the
bicepconfig.json
In this iteration we restrict the import gesture to fetch providers from OCI compliant registries alone. We plan to enhance the importing mechanism to use registry aliasing (similarly to Bicep module handling) so users can load provider definitions from offline sources such as disk in a separate proposal
- Support for third party extensibility providers
The design proposal can be extended to support third party extensibility providers that are composed from type definitions as well as a server-side component. That said, there is more work necessary to define exactly how the server-side component of an extensibility provider is packaged, how it's hosted and other details that should be elaborated and described in full as independent capabilities of the system. For this proposal we will defer this conversation and manage it as an incremental feature on top of this proposal.
- Signing of provider manifests
The current proposal doesn't address the signing of provider manifests. We expect to add signed manifests as an incremental feature that uses notary. We plan to address this in a separate proposal.
- Automated publishing of provider manifests to a registry
The current proposal doesn't address the automated publishing of provider manifests to a registry. We plan to manually publish the
az
provider manifest to the MCR registry as part of the initial implementation. We will address a process to publish manifests as a separate proposal in the future.Advantages of this proposal
Open Questions
The text was updated successfully, but these errors were encountered: