-
-
Notifications
You must be signed in to change notification settings - Fork 2
security best practices
title: Security best practices titleSuffix: Azure DevOps description: Best practices for managing security and keeping your data secure in Azure DevOps. ms.subservice: azure-devops-security ms.topic: best-practice ms.author: chcomley author: chcomley monikerRange: '<= azure-devops' ms.date: 09/18/2023
[!INCLUDE version-lt-eq-azure-devops]
This document is taken from https://learn.microsoft.com/en-us/azure/devops/organizations/security/security-best-practices?view=azure-devops and is updated to reflect how the rules from this repository indicated with ⭕ bold are related to the best practices in the original document.
When you're working with information and data, particularly in a cloud-based solution like Azure DevOps Services, prioritizing security should always be your primary concern. While Microsoft maintains the security of the underlying cloud infrastructure, it's your responsibility to configure security in Azure DevOps.
Although it's not mandatory, incorporating best practices while using Azure DevOps can enhance your experience and make it more secure. We've compiled the following best practices that aim to keep your Azure DevOps environment secure:
- Securing your Azure DevOps environment
- Restrict access through scoped permissions at the organization/collection, project, or object level
- Maintain tight control of administrators and security groups
- Scope service accounts and service connections
- Learn best practices for authenticating when integrating with Azure DevOps
- Secure specific product areas and associated services, like Azure Artifacts, Azure Boards, Azure Pipelines, Azure Repos, Azure Test Plans, and GitHub integrations.
- If your organization uses MSA accounts, then remove inactive users directly from the organization, as you have no other way to prevent access. When you do so, you can't create a query for work items assigned to the removed user account. For more information, see Delete users from Azure DevOps.
- If your organization is connected to Azure AD, then you can disable or delete the Azure AD user account and leave your Azure DevOps user account active. In this way, you can continue to query work item history using your Azure DevOps user ID.
- Revoke user PATs.
- Revoke any special permissions that may have been granted to individual user accounts.
- Reassign work from users you’re removing to current team members.
Integrate Azure DevOps with Azure AD to have a single plane for identity. Consistency and a single authoritative source increases clarity and reduces security risks from human errors and configuration complexity. The key to end governance is to have multiple role assignments (with different role definitions and different resource scopes to the same Azure AD groups). Without Azure AD, you're solely responsible for controlling organization access.
Using Azure AD also allows you to access other security features, like multi-factor authentication or other conditional access policies.
For more information, see the following articles:
- About accessing your organization with Azure AD
- Add AD/Azure AD users or groups to a built-in security groups
- Limit access by location or IP addresses
- Manage conditional access
- Require all users to use multi-factor authentication (MFA)
Once you have an Azure AD backed organization, you can turn on Auditing in your Security policies. Periodically review audit events to monitor and react to unexpected usage patterns by administrators and other users.
A few ways to do so might include:
- Set up an allowlist to restrict specific IPs.
- Always use encryption.
- Validate certificates.
- Implement Web application firewalls (WAFs), so they can filter, monitor, and block any malicious web-based traffic to and from Azure DevOps.
- For more information, see this guidance on application management best practices
The system manages permissions at different levels - individual, collection, project, and object - and assigns them to one or more built-in groups by default.
-
Only give users and services the minimum amount of access needed to perform their business functions.
-
Disable inheritance where possible. Due to the allow-by-default nature of inheritance, unexpected users can get access or permissions. For more information, read about inheritance.
⭕ Azure.DevOps.Pipelines.Core.InheritedPermissions ⭕ Azure.DevOps.Pipelines.Environments.InheritedPermissions ⭕ Azure.DevOps.Pipelines.Releases.Definition.InheritedPermissions ⭕ Azure.DevOps.Repos.InheritedPermissions ⭕ Azure.DevOps.ServiceConnections.InheritedPermissions ⭕ Azure.DevOps.Tasks.VariableGroup.InheritedPermissions
-
Learn more about permissions here:
- Limit access to projects and repos to reduce the risk of leaking sensitive information and deploying insecure code to production.
- Use either the built-in security groups or custom security groups to manage permissions. For more information, see Grant or restrict permissions to select tasks and project-scoped user groups.
- Enable the Limit user visibility for projects preview feature for the organization, which restricts access to only those projects that you add users to.
- Add users to the Project-scoped users group, so they can only see and select users and groups in the project that they're connected to from a people picker.
- Disable "Allow public projects" in your organization's policy settings to prevent every organization user from creating a public project. Azure DevOps Services allows you to change the visibility of your projects from public to private, and vice-versa. If users haven't signed into your organization, they have read-only access to your public projects. If users have signed in, they can be granted access to private projects and make any permitted changes to them.
- Don’t allow users to create new projects.
- Block external guest access entirely by disabling the "Allow invitations to be sent to any domain" policy. It's a good idea to do so if there's no business need for it.
- Use a different email or user principal name (UPN) for your personal and business accounts, even though it's allowed. This action eliminates the challenge of disambiguating between your business and personal accounts when the email/UPN is the same.
- Put all the external guest users in a single Azure AD group and manage the permissions of that group appropriately. You can easily manage and audit this way.
- Remove direct assignments so the group rules apply to those users. For more information, see Add a group rule to assign access levels.
- Reevaluate rules regularly on the Group rules tab of the Users page. Clarify whether any group membership changes in Azure AD might affect your organization. Azure AD can take up to 24 hours to update dynamic group membership. Every 24 hours and anytime a group rule changes, rules get automatically reevaluated in the system.
- For more information, see B2B guests in the Azure AD.
See the following recommendations for assigning permissions to security groups and users groups.
Do :::image type="icon" source="../../media/icons/checkmark.png" border="false"::: | Don't :::image type="icon" source="../../media/icons/delete-icon.png" border="false"::: |
---|---|
Use Azure Active Directory, Active Directory, or Windows security groups when you're managing lots of users. | Don’t change the default permissions for the Project Valid Users group. This group can access and view project information. ⭕ Azure.DevOps.`*.ProjectValidUsers |
When you're adding teams, consider what permissions you want to assign to team members who need to create and modify area paths, iteration paths, and queries. | Don't add users to multiple security groups that contain different permission levels. In certain cases, a Deny permission level may override an Allow permission level. |
When you're adding many teams, consider creating a Team Administrators custom group where you allocate a subset of the permissions available to Project Administrators. | Don't change the default assignments made to the Project Valid Users groups. If you remove or set View instance-level information to Deny for one of the Project Valid Users groups, no users in the group can access whatever project, collection, or deployment you set the permission on. ⭕ Azure.DevOps.`*.ProjectValidUsers |
Consider granting the work item query folders Contribute permission to users or groups who require the ability to create and share work item queries for the project. | Don't assign permissions that are noted as Assign only to service accounts to user accounts. |
Keep groups as small as possible. Access should be restricted, and the groups should be frequently audited. | |
Take advantage of built-in roles and default to Contributor for developers. Admins get assigned to the Project Administrator security group for elevated permissions, allowing them to configure security permissions. |
For more information, see Valid user groups.
You can change the configuration of your organization or project if you have Project Collection Administrator and Project Administrator access. To protect access to these built-in administrator groups, require just-in-time access using an Azure AD Privileged Identity Management (PIM) group.
Note
Make sure any user with elevated access using a PIM group also has standard access to the organization, so they can view the page to refresh their permissions.
- Activate your access.
- Refresh your permissions in Azure DevOps.
- Take the action requiring administrator access.
Note
Users have elevated access in Azure DevOps for up to 1 hour after their PIM group access gets deactivated.
- Ensure service accounts have zero interactive sign-in rights.
- Restrict service account privileges to the bare minimum necessary.
- Use a different identity for the report reader account, if you use domain accounts for your service accounts. For more information, see Service accounts and dependencies.
- Use local accounts for user accounts, if you're installing a component in a workgroup. For more information, see Service account requirements.
- Use service connections when possible. Service connections provide a secure mechanism to connect to assorted services without the need to pass in secret variables to the build directly. - Restrict these connections to the specific place they should be used and nothing more.
- Monitor service account activity and create audit streaming. Auditing allows you to detect and react to suspicious sign-ins and activity.
- For more information, see Common service connection types.
-
Scope Azure Resource Manager, and other service connections, only to the resources and groups to which they need access. Service connections shouldn't have broad contributor rights on the entire Azure subscription.
-
Don’t give users generic or broad contributor rights on the Azure subscription.
-
Don’t use Azure Classic service connections, as there’s no way to scope the permissions.
-
Make sure the resource group only contains Virtual Machines (VMs) or resources that the build needs access to.
-
Use a purpose-specific team service account to authenticate a service connection.
-
For more information, see Common service connection types.
Select your authentication methods from the following sources:
Explore alternatives like service principals and managed identities that enable you to use Azure AD tokens to access Azure DevOps resources. Such tokens carry less risk when leaked compared to PATs and contain benefits like easy credential management.
If possible, we recommended to always use identity services for authentication instead of cryptographic keys since managing keys securely with application code is challenging and can lead to mistakes like accidentally publishing sensitive access keys to public code repositories like GitHub. However, if you must use personal access tokens (PATs), consider the following guidelines:
-
PATs should always be scoped to specific roles.
-
PATs shouldn't provide global access to multiple organizations.
-
PATs shouldn't grant write or manage permissions on builds or releases.
-
PATs should have an expiration date and be kept secret since they're as critical as passwords.
-
PATs should never be hardcoded in the application code, even if it's tempting to do so to simplify the code.
-
Administrators should regularly audit all PATs using the REST APIs and revoke any that doesn't meet the above criteria.
-
Keep your PATs a secret. Your tokens are as critical as passwords.
-
Store your tokens in a safe place.
-
Don’t hard code tokens in applications. It can be tempting to simplify code to obtain a token for a long period of time and store it in your application, but don’t do that.
-
Give tokens an expiration date.
-
For more information, check out the following articles:
- Make sure you understand the difference between feeds, project, and project collection administrators. For more information, see Configure Azure Artifacts settings.
- For more information, see Set feed permissions.
- Review Configure and customize Azure Boards before you customize a process.
- See the following articles:
- Use extends templates.
- For more information about how to set permission levels for pipelines, see Set pipeline permissions.
-
Require at least one reviewer outside of the original requester. The approver shares coownership of the changes and should be held equally accountable for any potential impact.
-
Require CI build to pass. This requirement is useful for establishing baseline code quality, through code linting, unit tests, and security checks, like virus and credential scans.
-
Ensure that the original pull requester can’t approve the change.
-
Disallow completion of a PR (Pull Request), even if some reviewers vote to wait or reject.
-
Reset code reviewer votes when recent changes get pushed.
-
Lock down release pipelines by running them only on specific production branches.
-
Enable “Enforce settable at queue time for variables” in your organization’s pipeline settings.
⭕ Azure.DevOps.Pipelines.Settings.LimitSetVariablesAtQueueTime
-
Don’t allow “Let users override this value when running this pipeline,” for variables set in the editor.
- Grant permissions to the smallest possible number of accounts.
- Have the most restrictive firewall that leaves your agents usable.
- Update pools regularly to ensure the build fleet isn’t running vulnerable code that a malicious actor can exploit.
- Use a separate agent pool for build artifacts that get shipped or deployed to production.
- Segment “sensitive” pool from nonsensitive pools, and only allow the use of credentials in build definitions that are locked to that pool.
-
Manage pipeline definitions with YAML (Yet Another Markup Language). YAML is the preferred method for managing pipeline definitions, as it provides traceability for changes and can follow approval guidelines.
-
Secure the pipeline definition Edit access to the minimum number of accounts.
-
Include sanity checks for variables in build scripts. A sanity check can mitigate a command injection attack through the settable variables.
⭕ Azure.DevOps.Pipelines.Settings.SanitizeShellTaskArguments
-
Set as few build variables as possible to “Settable at release time.”
-
Avoid remotely fetched resources, but, if necessary, use versioning and hash checking.
-
Don’t log secrets.
-
Don’t store secrets in pipeline variables, use Azure KeyVault. Regularly scan your build pipelines to ensure secrets aren’t being stored in build pipeline variables.
⭕ Azure.DevOps.Tasks.VariableGroup.NoKeyVaultNoSecrets Azure.DevOps.Tasks.VariableGroup.NoPlainTextSecrets Azure.DevOps.Pipelines.Core.NoPlainTextSecrets Azure.DevOps.Pipelines.Releases.Definition.NoPlainTextSecrets
-
Don’t let users run builds against arbitrary branches or tags on security-critical pipelines.
-
Disable inheritance on the pipeline, as inherited permissions are broad and don’t accurately reflect your needs for permissions.
-
Limit job authorization scopes in all cases.
⭕ Azure.DevOps.Pipelines.Settings.LimitJobAuthorizationScope, Azure.DevOps.Pipelines.Settings.LimitJobAuthorizationScopeForReleasePipelines and Azure.DevOps.Pipelines.Settings.LimitJobAuthorizationScopeForYamlPipelines
-
Set the “Require a minimum number of reviewers,” policy to on, so that every pull request gets reviewed by at least two approvers.
-
Configure security policies specific to each repository or branch, instead of project wide. Security policies reduce risk, enforce change management standards, and improve your team’s quality of code.
-
Store production secrets in a separate KeyVault and ensure that access is only granted on a need-to-know basis to keep nonproduction builds separate.
-
Don’t mix test environments with production, including use of credentials.
-
Disable forking. The more forks there are, the harder it's to keep track of each fork’s security. Also, a user can easily fork a copy of a repository to their own private account.
-
Don't provide secrets to fork builds.
⭕ Azure.DevOps.Pipelines.Settings.RestrictSecretsForPullRequestFromFork
-
For Git, check your production build definitions in the project’s git repository, so they can be scanned for credentials.
-
Configure a branch control check so that only pipelines running in the context of the
production
branch may use theprod-connection
.⭕ Azure.DevOps.ServiceConnections.ProductionBranchLimit and Azure.DevOps.Pipelines.Environments.ProductionBranchLimit
-
For more information, see Other security considerations.
-
Improve code quality with branch policies. For more information about branch permissions and policies, see Set branch permissions.
- Disable Personal Access Token (PAT)-based authentication, so the OAuth flow gets used with the GitHub service connection.
- Never authenticate GitHub service connections as an identity that's an administrator or owner of any repositories.
- Never use a full-scope GitHub PAT (Personal Access Token) to authenticate GitHub service connections.
- Don't use a personal GitHub account as a service connection with Azure DevOps.