API implementation
Service client
Http pipeline
Since the client library generally wraps one or more HTTP requests, it’s important to support standard network capabilities. Although not widely understood, asynchronous programming techniques are essential in developing resilient web services. Many developers prefer synchronous method calls for their easy semantics when learning how to use a technology. The HTTP pipeline is a component in the azure-core
library that assists in providing connectivity to HTTP-based Azure services.
✅ DO use the HTTP pipeline to send requests to service REST endpoints.
☑️ YOU SHOULD include the following policies in the HTTP pipeline:
- Unique Request ID (
azure.core.pipeline.policies.RequestIdPolicy
) - Headers (
azure.core.pipeline.policies.HeadersPolicy
) - Telemetry (
azure.core.pipeline.policies.UserAgentPolicy
) - Proxy (
azure.core.pipeline.policies.ProxyPolicy
) - Content decoding (
azure.core.pipeline.policies.ContentDecodePolicy
) - Retry (
azure.core.pipeline.policies.RetryPolicy
andazure.core.pipeline.policies.AsyncRetryPolicy
) - Credentials (e.g.
BearerTokenCredentialPolicy
,AzureKeyCredentialPolicy
, etc) - Distributed tracing (
azure.core.pipeline.policies.DistributedTracingPolicy
) - Logging (
azure.core.pipeline.policies.NetworkTraceLoggingPolicy
)
from azure.core.pipeline import Pipeline
from azure.core.pipeline.policies import (
BearerTokenCredentialPolicy,
ContentDecodePolicy,
DistributedTracingPolicy,
HeadersPolicy,
HttpLoggingPolicy,
NetworkTraceLoggingPolicy,
UserAgentPolicy,
)
class ExampleClient(object):
...
def _create_pipeline(self, credential, base_url=None, **kwargs):
transport = kwargs.get('transport') or RequestsTransport(**kwargs)
try:
policies = kwargs['policies']
except KeyError:
scope = base_url.strip("/") + "/.default"
if hasattr(credential, "get_token"):
credential_policy = BearerTokenCredentialPolicy(credential, scope)
else:
raise ValueError(
"Please provide an instance from azure-identity or a class that implement the 'get_token protocol"
)
policies = [
HeadersPolicy(**kwargs),
UserAgentPolicy(**kwargs),
ContentDecodePolicy(**kwargs),
RetryPolicy(**kwargs),
credential_policy,
HttpLoggingPolicy(**kwargs),
DistributedTracingPolicy(**kwargs),
NetworkTraceLoggingPolicy(**kwargs)
]
return Pipeline(transport, policies)
Custom policies
Some services may require custom policies to be implemented. For example, custom policies may implement fall back to secondary endpoints during retry, request signing, or other specialized authentication techniques.
☑️ YOU SHOULD use the policy implementations in azure-core
whenever possible.
✅ DO review the proposed policy with the Azure SDK Architecture Board. There may already be an existing policy that can be modified/parameterized to satisfy your need.
✅ DO derive from HTTPPolicy/AsyncHTTPPolicy (if you need to make network calls) or SansIOHTTPPolicy (if you do not).
✅ DO ensure thread-safety for custom policies. A practical consequence of this is that you should keep any per-request or connection bookkeeping data in the context rather than in the policy instance itself.
✅ DO document any custom policies in your package. The documentation should make it clear how a user of your library is supposed to use the policy.
✅ DO add the policies to the azure.<package name>.pipeline.policies
namespace.
Service Methods
Parameter validation
⛔️ DO NOT use isinstance
to validate parameter value types other than for built-in types (e.g. str
etc). For other types, use structural type checking.
Supporting types
✅ DO implement __repr__
for model types. The representation must include the type name and any key properties (that is, properties that help identify the model instance).
✅ DO truncate the output of __repr__
after 1024 characters.
Extensible enumerations
Any Enums defined in the SDK should be interchangable with case-insensitive strings. This is achieved by using the CaseInsensitiveEnumMeta
class defined in azure-core
.
from enum import Enum
from six import with_metaclass
from azure.core import CaseInsensitiveEnumMeta
class MyCustomEnum(with_metaclass(CaseInsensitiveEnumMeta, str, Enum)):
FOO = 'foo'
BAR = 'bar'
SDK Feature implementation
Configuration
✅ DO honor the following environment variables for global configuration settings:
환경 변수 | 목적 |
---|---|
프록시 설정 | |
HTTP_PROXY | HTTP 연결을 위한 프록시 |
HTTPS_PROXY | HTTPS 연결을 위한 프록시 |
NO_PROXY | 프록시를 사용하면 안 되는 호스트 |
ALL_PROXY | HTTP_PROXY 그리고/또는 HTTPS_PROXY가 정의되지 않은 경우 HTTP 그리고/또는 HTTPS 연결을 위한 프록시 |
ID | |
MSI_ENDPOINT | Azure AD MSI 자격 증명 |
MSI_SECRET | Azure AD MSI 자격 증명 |
AZURE_USERNAME | U/P 인증을 위한 Azure 사용자 이름 |
AZURE_PASSWORD | U/P 인증을 위한 Azure 비밀 번호 |
AZURE_CLIENT_CERTIFICATE_PATH | Azure Active Directory |
AZURE_CLIENT_ID | Azure Active Directory |
AZURE_CLIENT_SECRET | Azure Active Directory |
AZURE_TENANT_ID | Azure Active Directory |
AZURE_AUTHORITY_HOST | Azure Active Directory |
파이프라인 구성 | |
AZURE_TELEMETRY_DISABLED | 원격 분석 비활성화 |
AZURE_LOG_LEVEL | 로그 레벨을 설정하여 로깅 활성화 |
AZURE_TRACING_DISABLED | 추적 비활성화 |
범용 SDK 구성 | |
AZURE_CLOUD | 소버린 클라우드 이름 |
AZURE_SUBSCRIPTION_ID | Azure 구독 |
AZURE_RESOURCE_GROUP | Azure 리소스 그룹 |
Logging
✅ DO use Pythons standard logging module.
✅ DO provide a named logger for your library.
The logger for your package must use the name of the module. The library may provide additional child loggers. If child loggers are provided, document them.
For example:
- Package name:
azure-someservice
- Module name:
azure.someservice
- Logger name:
azure.someservice
- Child logger:
azure.someservice.achild
These naming rules allow the consumer to enable logging for all Azure libraries, a specific client library, or a subset of a client library.
✅ DO use the ERROR
logging level for failures where it’s unlikely the application will recover (for example, out of memory).
✅ DO use the WARNING
logging level when a function fails to perform its intended task. The function should also raise an exception.
Don’t include occurrences of self-healing events (for example, when a request will be automatically retried).
✅ DO use the INFO
logging level when a function operates normally.
✅ DO use the DEBUG
logging level for detailed trouble shooting scenarios.
The DEBUG
logging level is intended for developers or system administrators to diagnose specific failures.
⛔️ DO NOT send sensitive information in log levels other than DEBUG
. For example, redact or remove account keys when logging headers.
✅ DO log the request line, response line, and headers for an outgoing request as an INFO
message.
✅ DO log an INFO
message, if a service call is canceled.
✅ DO log exceptions thrown as a WARNING
level message. If the log level set to DEBUG
, append stack trace information to the message.
You can determine the logging level for a given logger by calling logging.Logger.isEnabledFor
.
Distributed tracing
✅ DO create a new trace span for each library method invocation. The easiest way to do so is by adding the distributed tracing decorator from azure.core.tracing
.
✅ DO use <package name>/<method name>
as the name of the span.
✅ DO create a new span for each outgoing network call. If using the HTTP pipeline, the new span is created for you.
✅ DO propagate tracing context on each outgoing service request.
Telemetry
Client library usage telemetry is used by service teams (not consumers) to monitor what SDK language, client library version, and language/platform info a client is using to call into their service. Clients can prepend additional information indicating the name and version of the client application.
✅ DO send telemetry information in the [User-Agent header] using the following format:
[<application_id> ]azsdk-python-<package_name>/<package_version> <platform_info>
<application_id>
: optional application-specific string. May contain a slash, but must not contain a space. The string is supplied by the user of the client library, e.g. “AzCopy/10.0.4-Preview”<package_name>
: client library (distribution) package name as it appears to the developer, replacing slashes with dashes and removing the Azure indicator. For example, “azure-keyvault-secrets” would specify “azsdk-python-keyvault-secrets”.<package_version>
: the version of the package. Note: this is not the version of the service<platform_info>
: information about the currently executing language runtime and OS, e.g. “Python/3.8.4 (Windows-10-10.0.19041-SP0)”
For example, if we re-wrote AzCopy
in Python using the Azure Blob Storage client library, we may end up with the following user-agent strings:
- (Python)
AzCopy/10.0.4-Preview azsdk-python-storage/4.0.0 Python/3.7.3 (Ubuntu; Linux x86_64; rv:34.0)
The azure.core.pipeline.policies.UserAgentPolicy
will provide this functionality if added to the HttpPipeline.
☑️ YOU SHOULD send additional (dynamic) telemetry information as a semi-colon separated set of key-value types in the X-MS-AZSDK-Telemetry
header. For example:
X-MS-AZSDK-Telemetry: class=BlobClient;method=DownloadFile;blobType=Block
The content of the header is a semi-colon key=value list. The following keys have specific meaning:
class
is the name of the type within the client library that the consumer called to trigger the network operation.method
is the name of the method within the client library type that the consumer called to trigger the network operation.
Any other keys that are used should be common across all client libraries for a specific service. DO NOT include personally identifiable information (even encoded) in this header. Services need to configure log gathering to capture the X-MS-SDK-Telemetry
header in such a way that it can be queried through normal analytics systems.
Considerations for clients not using the UserAgentPolicy from azure-core
✅ DO allow the consumer of the library to set the application ID by passing in an application_id
parameter to the service client constructor. This allows the consumer to obtain cross-service telemetry for their app.
✅ DO enforce that the application ID is no more than 24 characters in length. Shorter application IDs allows service teams to include diagnostic information in the “platform information” section of the user agent, while still allowing the consumer to obtain telemetry information for their own application.
Testing
✅ DO use pytest as the test framework.
☑️ YOU SHOULD use pytest-asyncio for testing of async code.
✅ DO make your scenario tests runnable against live services. Strongly consider using the Python Azure-DevTools package for scenario tests.
✅ DO provide recordings to allow running tests offline/without an Azure subscription
✅ DO support simultaneous test runs in the same subscription.
✅ DO make each test case independent of other tests.
Code Analysis and Style Tools
✅ DO use pylint for your code. Use the pylintrc file in the root of the repository.
✅ DO use flake8-docstrings to verify doc comments.
✅ DO use Black for formatting your code.
☑️ YOU SHOULD use MyPy to statically check the public surface area of your library.
You don’t need to check non-shipping code such as tests.
Making use of Azure Core
The azure-core
package provides common functionality for client libraries. Documentation and usage examples can be found in the azure/azure-sdk-for-python repository.
HTTP pipeline
The HTTP pipeline is an HTTP transport that is wrapped by multiple policies. Each policy is a control point that can modify either the request or response. A default set of policies is provided to standardize how client libraries interact with Azure services.
For more information on the Python implementation of the pipeline, see the documentation.
Protocols
Many of the protocols mandated by the design guidelines have default implementations in azure-core
.
LROPoller
T = TypeVar("T")
class LROPoller(Protocol):
def result(self, timeout=None) -> T:
""" Retrieve the final result of the long running operation.
:param timeout: How long to wait for operation to complete (in seconds). If not specified, there is no timeout.
:raises TimeoutException: If the operation has not completed before it timed out.
"""
...
def wait(self, timeout=None) -> None:
""" Wait for the operation to complete.
:param timeout: How long to wait for operation to complete (in seconds). If not specified, there is no timeout.
"""
def done(self) -> boolean:
""" Check if long running operation has completed.
"""
def add_done_callback(self, func) -> None:
""" Register callback to be invoked when operation completes.
:param func: Callable that will be called with the eventual result ('T') of the operation.
"""
...
azure.core.polling.LROPoller
implements the LROPoller
protocol.
ItemPaged
T = TypeVar("T")
class ByPagePaged(Protocol, Iterable[Iterable[T]]):
continuation_token: "str"
class ItemPaged(Protocol, Iterable[T]):
continuation_token: "str"
def by_page(self) -> ByPagePaged[T] ...
azure.core.ItemPaged
implements the ItemPaged
protocol.
See the ItemPaged protocol for additional information.
DiagnosticsResponseHook
class ResponseHook(Protocol):
__call__(self, headers, deserialized_response): -> None ...
Python language and code style
✅ DO follow the general guidelines in PEP8 unless explicitly overridden in this document.
⛔️ DO NOT “borrow” coding paradigms from other languages.
For example, no matter how common Reactive programming is in the Java community, it’s still unfamiliar for most Python developers.
✅ DO favor consistency with other Python components over other libraries for the same service.
It’s more likely that a developer will use many different libraries using the same language than a developer will use the same service from many different languages.
Error handling
✅ DO use exception chaining to include the original source of the error when catching and raising new exceptions.
# Yes:
try:
# do something
something()
except:
# __context__ will be set correctly
raise MyOwnErrorWithNoContext()
# No:
success = True
try:
# do something
something()
except:
success = False
if not success:
# __context__ is lost...
raise MyOwnErrorWithNoContext()
Naming conventions
✅ DO use snake_case for variable, function, and method names:
# Yes:
service_client = ServiceClient()
service_client.list_things()
def do_something():
...
# No:
serviceClient = ServiceClient()
service_client.listThings()
def DoSomething():
...
✅ DO use Pascal case for types:
# Yes:
class ThisIsCorrect(object):
pass
# No:
class this_is_not_correct(object):
pass
# No:
class camelCasedTypeName(object):
pass
✅ DO use ALL CAPS for constants:
# Yes:
MAX_SIZE = 4711
# No:
max_size = 4711
# No:
MaxSize = 4711
✅ DO use snake_case for module names.
Method signatures
⛔️ DO NOT use static methods (staticmethod
). Prefer module level functions instead.
Static methods are rare and usually forced by other libraries.
⛔️ DO NOT use simple getter and setter functions. Use properties instead.
# Yes
class GoodThing(object):
@property
def something(self):
""" Example of a good read-only property."""
return self._something
# No
class BadThing(object):
def get_something(self):
""" Example of a bad 'getter' style method."""
return self._something
⚠️ YOU SHOULD NOT have methods that require more than five positional parameters. Optional/flag parameters can be accepted using keyword-only arguments, or **kwargs
.
See TODO: insert link for general guidance on positional vs. optional parameters here.
✅ DO use keyword-only arguments for optional or less-often-used arguments for modules that only need to support Python 3.
# Yes
def foo(a, b, *, c, d=None):
# Note that I can even have required keyword-only arguments...
...
✅ DO use keyword-only arguments for arguments that have no obvious ordering.
# Yes - `source` and `dest` have logical order, `recurse` and `overwrite` do not.
def copy(source, dest, *, recurse=False, overwrite=False) ...
# No
def copy(source, dest, recurse=False, overwrite=False) ...
✅ DO specify the parameter name when calling methods with more than two required positional parameters.
def foo(a, b, c):
pass
def bar(d, e):
pass
# Yes:
foo(a=1, b=2, c=3)
bar(1, 2)
bar(e=3, d=4)
# No:
foo(1, 2, 3)
✅ DO specify the parameter name for optional parameters when calling functions.
def foo(a, b=1, c=None):
pass
# Yes:
foo(1, b=2, c=3)
# No:
foo(1, 2, 3)
Public vs “private”
✅ DO use a single leading underscore to indicate that a name isn’t part of the public API. Non-public APIs aren’t guaranteed to be stable.
⛔️ DO NOT use leading double underscore prefixed method names unless name clashes in the inheritance hierarchy are likely. Name clashes are rare.
✅ DO add public methods and types to the module’s __all__
attribute.
✅ DO use a leading underscore for internal modules. You may omit a leading underscore if the module is a submodule of an internal module.
# Yes:
azure.exampleservice._some_internal_module
# Yes - some_internal_module is still considered internal since it is a submodule of an internal module:
azure.exampleservice._internal.some_internal_module
# No - some_internal_module is considered public:
azure.exampleservice.some_internal_module
Types (or not)
✅ DO prefer structural subtyping and protocols over explicit type checks.
✅ DO derive from the abstract collections base classes collections.abc
(or collections
for Python 2.7) to provide custom mapping types.
✅ DO provide type hints PEP484 for publicly documented classes and functions.
- See the suggested syntax for Python 2.7 and 2.7-3.x straddling code for guidance for Python 2.7 compatible code. Do not do this for code that is Python 3 specific (e.g.
async
clients.)
Threading
✅ DO maintain thread affinity for user-provided callbacks unless explicitly documented to not do so.
✅ DO explicitly include the fact that a method (function/class) is thread safe in its documentation.
Examples: asyncio.loop.call_soon_threadsafe
, queue
☑️ YOU SHOULD allow callers to pass in an Executor
instance rather than defining your own thread or process management for parallelism.
You may do your own thread management if the thread isn’t exposed to the caller in any way. For example, the LROPoller
implementation uses a background poller thread.