Identity Platform
Not Live Yet
You are viewing a preview of documentation for the future state of GitLab Identity v3 (mid 2024). See the Access Management Policy for the GitLab Identity v2 current state with baseline entitlements and access requests. See the roadmap in the epics gantt chart.Identity Platform Architecture
- CI/CD Pipeline
- Access Requests
- Approvals
- Access Check (accesschk) Audit
- Terraform GitOps Configuration-as-Code
User Guides
- Admin Guide
- Application Configuration Guide
- Audit and Compliance Guide
- Change Management Guide
- Developer Guide
- Incident Response Guide
- Manager Guide
- Offboarding Ops Guide
- Onboarding Ops Guide
- Policy Management Guide
- Team Member End User Guide
Why It Exists
We use Terraform with a GitOps CI/CD pipeline for managing the configuration and resources in each platform/vendor (ex. Okta Settings and Okta Applications), and specifying which groups are assigned to each application/resource/permission in each respective platform (ex. Okta application groups, GCP project roles, etc). We believe that configuration and state management auditlogs are much easier to audit without all of the user membership changes, so we use Access Control to keep those changes separate and provide improved automation user experience for those use cases.
In other words, it’s not realistic to have every user perform self-service Terraform changes for access requests.
The Identity Engineering team has created a standardized group and user management architecture that is used for role-based access control. Each of the standardized roles are used in combination with Groups in various systems to uniformally assign access to a known list of users that are managed with centralized policies and change management processes.
The Identity Platform is designed to use the API for each of the vendors and provide the integration plumbing between them using policy files and manifests that are stored in a centrally managed GitLab repository. Since we maintain a centralized database (via Git), we can provide a unified policy management, streamlined user experience, and provide 90%+ automation for onboarding, access requests, career mobility, and offboarding.
While many vendors have partial capabilities and some overlap, the ease-of-use and integrations vary drastically and we have had a lot of inefficiency problems in Identity v2. Since Group management is at the heart of the problem, the Identity Platform is primarily focused on group user management automation that is used downstream in vendor configurations that we automate using Terraform where possible.
Open Source
Although the risks that we need to solve are confidential, we believe in open sourcing the best practice documentation and tools to solve those risks after we have mitigated the risk to give back to the community and inspire other companies.
We are stronger and more secure together.
Our Identity Platform open source projects can be explored at https://gitlab.com/gitlab-identity. Our team members also maintain side project packages that are useful dependencies and development helpers at https://gitlab.com/provisionesta.
Terminology
Identity Types
Okta User Attribute:
rbac_type
We have different categories of users (ex. employees, contractors, service accounts, system administrators) that need birthright access to various systems. We use our access level wristbands color coding to specify each Okta user’s type as blue
, purple
, gray
, brown
, or black
.
Identity Roles
Okta User Attribute:
rbac_role
An Identity Role is a standardized snake case format for the functional team that the user is on or job title, as it relates to access control and permissions.
This is a next generation iteration of a GitLab Identity v2 job family and role-based baseline entitlements.
Each role uses a syntax of {department_slug}_{functional_team_slug}_{specific_role_if_applicable}
. For example, if your department is Infrastructure
, the RBAC slug is infra
.
The job title of Site Reliability Engineer
is too vague for IAM/RBAC purposes since we have multiple functional teams of Site Reliability Engineers that should have different levels of access.
We have defined RBAC slugs for each functional team in Infrastructure that include the infra_
prefix:
infra_analytics
infra_people_leader
infra_platforms_leader
infra_scalability_leader
infra_eng_leader
infra_ded_general
infra_ded_env_automation
infra_ded_switchboard
infra_ded_us_pubsec
infra_ded_ext
infra_deployment
infra_foundations
infra_observability
infra_ops
infra_orchestration
infra_practices
Each of these functional teams usually has 1-10(ish) team members that report to one or two managers. Some functional teams may be larger (ex. product manager, sales roles, etc).
From a policy perspective, we associate a user with a role based on their manager
in most cases, or a specific job title
or a named user handle
if needed.
There is no need for a specific role for most of these teams (no IAM difference between the Site Reliability Engineer title and Engineering Manager title). For teams that have approval permission requirements, we will suffix _engineer
, _manager
, _leader
, etc as appropriate. We may also suffix the region for certain roles (ex. sales_ent_amer_east_northeast
). For roles that aren’t necessarily on a specific team, _general
(Staff/Principal Engineers across multiple functions), _ext
(contractors with least privilege access), and others at the Identity Team’s naming convention discretion.
The team manager and department leadership is responsible for naming their functional teams. The Identity team is responsible for parsing the handbook pages or provided organization chart documentation for each team and creating the short hand standardized abbreviation nomenclature.
As our organization structure ebbs and flows, we have 200-250 unique Identity Roles at any given time. See all of the roles in the policies and manifests repositories.
Division/Function | Identity Roles | Upstream Data Source |
---|---|---|
Administrators | Policies | Identity Engineering |
Executive | Policies | Identity Engineering |
Finance | Policies | Managers and Job Titles |
Legal | Policies | Identity Engineering |
Marketing | Policies | Department Names |
People | Policies | Handbook Page and Manager |
Sales | Policies | Sales EBA Team and Department/Manager |
Product Development | Policies | DevOps Stages / YAML |
Product Production | Policies | Handbook Page |
Security | Policies | Handbook Page |
Service Accounts | Policies | Identity Engineering |
Identity Organization Units
The goal of creating Identity Roles is to implement role-based access control (RBAC) for application and group assignment, however it can be tedious and inefficient to add every single role to each application.
For less granular permissions or nonsensitive access, we can grant access to an organization unit instead of a long list of roles.
An Identity Organization Unit (OU) is a grouping of two or more Identity Roles for higher-level groups that are useful for granting access to entire divisions, departments, sub-departments, teams, regional groups or other functional groups. The Identity team is responsible for organization unit conventions with collaborative input from the Director/VP/EBA team members for each respective division.
A user can only belong to one role, however they can belong to multiple organization units.
See all of the organization units in the policies and manifests repositories.
Identity Groups
We use Identity Types, Identity Roles and Identity Organization Units across multiple systems for role-based access control (RBAC) by creating an Okta Group, Google Group, and GitLab SaaS Group with the same name. Each of these groups can be attached to one or more Okta Applications, Google Cloud projects or folders, or GitLab Groups or Projects. Additional systems may be added in the future.
Since the user list for each role is centrally managed and users are automatically synced, this removes a lot of manual provisioning inefficiency.
Our automation automatically creates each group and keeps the user list synced every hour with audit logs and automation workflows for any detected attribute and policy manifest changes. See more about manifests, auditlog, and provisioning.
The generated Okta Groups and Google Groups can be assigned to applications, accessed in scheduled generation of JSON/YAML/CSV, accessed via API calls to the Identity Platform, audited by Compliance team members, viewed in generated Google Sheets, and can be trusted as a single source of truth that is updated hourly or daily.
Okta and Google Groups
Each Type group is prefixed with rbac_type_{color}
.
Each Organization Unit group is prefixed with rbac_ou_{ou_name}
Each Role group is prefixed with rbac_role_{color}_{role_name}
.
We do not create a role group for brown
accounts since each service account has least privilege permissions based on the use case. The grouping of all service accounts uses the rbac_type_brown
group.
GitLab SaaS Groups
The GitLab SaaS group for the type, role, or organization unit can be invited as a Direct Member on any other GitLab group (with child projects) or a GitLab project with the appropriate permission role.
@gitlab-rbac/type/{color}
@gitlab-rbac/ou/{ou_name}
@gitlab-rbac/role/{role_name}
This allows us to grant minimum permissions at the top level of the gitlab-com
and gitlab-org
namespaces, and programatically add elevated Developer
, Maintainer
, and Owner
permissions in downstream groups and projects for specific teams to self manage their own child namespaces without having full read-write access across all projects inherited by the top-level namespace.
In the (distant) future, we will manage GitLab groups and project configuration settings using Terraform.
Identity Policies
Each Identity Role and Identity Organization Unit is defined in a YAML file with a list of conditional rules that defines which user attributes need to match for a user to be added to the manifest.
For granular roles, you can specify one or more job titles. If a job title exists in multiple departments, you can specify job titles in a specific department or that report to a specific manager. For less granular roles, you can use a user’s manager since one manager usually owns a specific named function. For larger functional teams, multiple managers can be mapped to a single role.
See all of the policies in the policies repository.
Identity Policy Rulesets
Our rules logic is simple, yet reasonably powerful. You can define one or more attributes to check for, and if any of the attribute conditions match, the user is added to that group.
For programmatic consistency with YAML, all attribute values are converted to lower_snake_case for calculation purposes. This removes all punctuation, replaces spaces with underscores, and solves for slight naming inconsistencies.
Infrastructure => infrastructure
Site Reliability Engineer => site_reliability_engineer
Director, Infrastructure => director_infrastructure
Available Attributes
You can use any of the Okta user profile attributes that were imported during the Manifest Users stage.
# {accessctl_key}: '{oktaApiKey}'
email_to_handle:
handle: 'email'
manager: 'managerEmail'
snake_case:
cost_center: 'costCenter'
division: 'division'
department: 'department'
management_level: 'workday_managementLevel'
organization_name: 'organization'
region: 'workday_region'
role: 'rbac_role'
title: 'title'
Single Attribute Rule
This covers the basic use case of matching the value of a single attribute. If any condition matches, the user will be added to the manifest.
# Any user with these job titles
accounting_payable_analyst:
- title: accounts_payable_analyst
- title: senior_accounts_payable_analyst
# Any user with these job titles
product_manager:
- title: product_manager
- title: senior_product_manager
- title: principal_product_manager
- title: group_manager_product
# Any user that reports to this manager, including the manager too
eng_productivity:
- manager: dmurphy
- handle: dmurphy
Multiple Attribute Rule
Since the goal of Identity Groups is to provide granularity, you can define multiple attributes to check (programmatically unlimited, but realistically less than 5). All attribute values must match. To perform an “either/or” rule, you would define an additional rule for each “or”.
# Any user in this specific department with this job title
dev_eng_leader:
- {department: development, title: distinguished_engineer}
- {department: development, title: senior_distinguished_engineer}
- {department: development, title: engineering_fellow}
- {department: development, title: principal_engineer}
- {department: development, title: principal_fullstack_engineer}
# Any user in this department, that is at a specific management level (to accommodate title variations)
dev_people_leader:
- {department: development, management_level: vice_president}
- {department: development, management_level: director}
Organization Unit Rules
Each Identity Role policy allows you to use the Okta user profile attribute keys. Organization Unit policies allow you to specify roles as well.
Although it is a best practice to exclusively use roles, we have the flexibility to use attributes or named user handles if needed.
infra_saas_prod_log_viewers:
- department: infrastructure
- handle: dmurphy
- handle: klibby
- role: sec_logging
- role: sec_sirt
- role: eng_core_database
- role: eng_core_dbre
infra_saas_extended:
- {department: infrastructure, management_level: leader}
- {department: infrastructure, management_level: vice_president}
- {division: engineering, management_level: leader}
- {division: engineering, management_level: vice_president}
- handle: dmurphy
- handle: klibby
- role: infra_saas_leaders
- role: infra_saas_scalability_leaders
- role: infra_saas_dedicated_general
- role: infra_saas_dedicated_env_automation
- role: infra_saas_dedicated_switchboard
- role: infra_saas_dedicated_us_pubsec
- role: infra_saas_deployment
- role: infra_saas_foundations
- role: infra_saas_observability
- role: infra_saas_ops
- role: infra_saas_orchestration
- role: infra_saas_practices
Manifest Generation
When the Identity Platform CI/CD job runs, it will calculate a fresh list of users that match the rules.
Each rule is evaluated independently, and all user email handles that match that attribute criteria are included in the returned array.
After all rules have been evaluated, the result arrays from each rule are aggregated and any duplicates are removed if a user matched two or more rules.
Generate a JSON manifest of users based on a policy.yml definition file
Manifest Does Not Exist Yet rbac/roles/{role_name}/manifest.json
[member] abarton@example.com Added user to manifest
[member] acummerata@example.com Added user to manifest
[member] akonopelski@example.com Added user to manifest
[member] alegros@example.com Added user to manifest
[member] arunte@example.com Added user to manifest
... truncated
[member] Added Total 119 users
File Created rbac/roles/{role_name}/manifest.json
Diff and Drift Management
We have diff detection for the previous and latest manifest for logging and alerting.
Generate a JSON manifest of users based on a policy.yml definition file
[member] acummerata@example.com Removed user from manifest
[member] akonopelski@example.com Removed user from manifest
[member] Removed Total 2 users
File Updated rbac/roles/{role_name}/manifest.json
Log Schema
{
"message": "User added to manifest",
"job_batch_id": "{uuid}",
"timestamp": "YYYY-MM-DDTHH:II:SS.000000Z",
"event": "accessctl.manifest.create.user.added",
"policy_type": "role",
"policy_name": "{role_name}",
"manifest_email": "acummerata@example.com",
"user_key": "acummerata",
"status": "active",
"status_updated_at_timestamp": "YYYY-MM-DDTHH:II:SS.000000Z",
"department": "entr",
"division": "sales",
"title": "inside_sales_representative",
"management_level": "individual_contributor",
"manager": "vdooley",
"region": "japac",
"rbac_type": null,
"rbac_role": "{role_name}"
},
{
"message": "User removed from manifest",
"job_batch_id": "{uuid}",
"timestamp": "YYYY-MM-DDTHH:II:SS.000000Z",
"event": "accessctl.manifest.create.user.removed",
"policy_type": "role",
"policy_name": "{role_name}",
"manifest_email": "akonopelski@example.com",
"user_key": null,
"status": "deprovisioned",
"status_updated_at_iso8601": "YYYY-MM-DDTHH:II:SS.000000Z"
}
Group User Sync
Many vendors rules engines work well for adding a user to a group, but there are limitations with removing a user from the group. This problem is amplified with offboarded users since deactivated users aren’t removed from the groups automatically (their authentication is blocked, however their authorization is still in-tact).
There is also limited to no functionality available with vendors to detecting changes and diff logging or triggering automation workflows.
With accessctl
, we have advanced change/diff audit logging and workflow automation, and keep groups in sync for adding and removing users.
For any Google Groups, Okta Groups, or tech stack APIs that are managed by the Identity Platform, our CI/CD jobs query the API to get all current members for that group, compare it to our manifest, and perform add user or remove user API calls if needed.
Adding Users to Group
// https://developer.okta.com/docs/reference/api/groups/#add-user-to-group
$okta_response = OktaApiClient::post(
connection: config('connections.okta.prod'),
uri: 'groups/ ' . $okta_group_id . '/members/' . $okta_user_id,
);
// https://developers.google.com/admin-sdk/directory/reference/rest/v1/members/insert
$google_response = GoogleApiClient::post(
connection: config('connections.google.workspace'),
url: 'https://admin.googleapis.com/admin/directory/v1/groups/' . $google_group_id . '/members',
form_data: [
'email' => $email,
'role' => 'MEMBER',
'type' => 'USER',
]
);
// https://docs.gitlab.com/ee/api/members.html#add-a-member-to-a-group-or-project
// https://docs.gitlab.com/ee/api/members.html#roles
$gitlab_response = GitlabApiClient::post(
connection: config('connections.gitlab.saas'),
uri: 'groups/' . $gitlab_group_id . '/members',
request_data: [
'user_id' => $gitlab_user_id,
'access_level' => $gitlab_access_level,
'expires_at' => null
]
);
Removing Users from Group
// https://developer.okta.com/docs/reference/api/groups/#remove-user-from-group
$okta_response = OktaApiClient::delete(
connection: config('connections.okta.prod'),
uri: 'groups/ ' . $okta_group_id . '/members/' . $okta_user_id,
);
// https://developers.google.com/admin-sdk/directory/reference/rest/v1/members/delete
$google_response = GoogleApiClient::delete(
connection: config('connections.google.workspace'),
url: 'https://admin.googleapis.com/admin/directory/v1/groups/' . $google_group_id . '/members/' . $email
);
// https://docs.gitlab.com/ee/api/members.html#remove-a-member-from-a-group-or-project
$gitlab_response = GitlabApiClient::delete(
connection: config('connections.gitlab.saas'),
uri: 'groups/' . $gitlab_group_id . '/members/' . $gitlab_user_id
);
CI/CD Pipeline Jobs
We consider Workday to be the source of truth for new users and changes to user profile attribute values, and we fetch those values from the Okta API Users endpoint. When a user’s Okta status changes to deprovisioned
, we consider them to be offboarded and remove them from all manifests.
The Identity Platform runs scheduled CI/CD pipeline jobs that use accessctl
CLI jobs to recalculate the list of users (a.k.a. manifest) for every Identity Role and Identity Organization Unit based on the latest data.
When a user attribute value changes in Workday and pushes to Okta, the Identity Platform will detect the diff during the manifests pipeline stage and create audit log entries and dispatch automation workflows in the auditlog pipeline stage. After a manifest has been created, the provisioning pipeline syncs all of the groups and group users across the rbac_*
prefixed Okta groups, Google Groups, @gitlab-rbac/*
GitLab groups that the Identity Platform manages.
When we reach maturity, the scheduled jobs will run every hour and all attribute values and groups will be automatically calculated and programatically maintained without any human user intervention or day-to-day provisioning tasks. While we are in PoC/alpha/beta, the jobs run daily and created in a merge request that is reviewed by the Identity Ops team for accuracy to catch any bugs or unintended changes.
graph LR subgraph accessctl GitLab CI/CD Pipeline Jobs direction LR subgraph Manifest Stage direction LR CI_MANIFEST_USER_JOB[["Stage 1.1<br />Users Job<br/>CLI manifest:users"]] CI_MANIFEST_ROLE_JOB[["Stage 1.2<br />Roles Job<br />CLI manifest:roles"]] CI_MANIFEST_GROUP_JOB[["Stage 1.3<br />Org Units Job<br/>CLI manifest:ou"]] end subgraph Auditlog Stage direction LR CI_CHANGELOG_USER_JOB[["Stage 2.1<br />Users Job<br/>CLI auditlog:users"]] CI_CHANGELOG_ATTRIBUTE_JOB[["Stage 2.2<br />Attributes Job<br/>CLI auditlog:attributes"]] CI_CHANGELOG_ROLE_JOB[["Stage 2.3<br />Roles Job<br/>CLI auditlog:roles"]] CI_CHANGELOG_GROUP_JOB[["Stage 2.4<br />Org Units Job<br/>CLI auditlog:ou"]] end subgraph Provisioning Stage direction LR subgraph Okta API CI_PROVISIONING_OKTA_USER_JOB["Stage 3.1<br />Okta Users Job<br />provision:okta-users"] CI_PROVISIONING_OKTA_GROUP_JOB["Stage 3.2<br />Okta Groups Job<br />provision:okta-groups"] end subgraph Google Workspace Directory API CI_PROVISIONING_GOOGLE_GROUP_JOB["Stage 3.3<br />Google Groups Job<br />provision:google-groups"] end subgraph GitLab.com SaaS API CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB["Stage 3.4<br />GitLab SaaS Groups Job<br />provision:gitlab-saas-groups"] end subgraph GitLab Self-Managed Instance API CI_PROVISIONING_GITLAB_SELF_GROUP_JOB["Stage 3.5<br />GitLab Self-Managed Groups Job<br />provision:gitlab-self-groups"] end end CI_MANIFEST_USER_JOB --> CI_MANIFEST_ROLE_JOB --> CI_MANIFEST_GROUP_JOB CI_CHANGELOG_USER_JOB --> CI_CHANGELOG_ATTRIBUTE_JOB --> CI_CHANGELOG_ROLE_JOB --> CI_CHANGELOG_GROUP_JOB CI_PROVISIONING_OKTA_USER_JOB --> CI_PROVISIONING_OKTA_GROUP_JOB --> CI_PROVISIONING_GOOGLE_GROUP_JOB --> CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB --> CI_PROVISIONING_GITLAB_SELF_GROUP_JOB click CI_MANIFEST_USER_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_MANIFEST_ROLE_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_MANIFEST_GROUP_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_CHANGELOG_USER_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_ATTRIBUTE_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_ROLE_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_GROUP_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_PROVISIONING_OKTA_USER_JOB "/handbook/security/identity/platform/provisioning/okta" "View Details" click CI_PROVISIONING_OKTA_GROUP_JOB "/handbook/security/identity/platform/provisioning/okta" "View Details" click CI_PROVISIONING_GOOGLE_GROUP_JOB "/handbook/security/identity/platform/provisioning/google" "View Details" click CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB "/handbook/security/identity/platform/provisioning/gitlab" "View Details" click CI_PROVISIONING_GITLAB_SELF_GROUP_JOB "/handbook/security/identity/platform/provisioning/gitlab" "View Details" classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px; end
Configuration and Data Repositories
The tf_*
Terraform GitOps repository contains the Terraform that manages the configuration and resources in each platform/vendor (ex. Okta Settings and Okta Applications), and specifies which groups are assigned to each application/resource/permission in each respective platform (ex. Okta application groups, GCP project roles, etc). We believe that configuration and state management auditlogs are much easier to audit without all of the user membership changes, so we use Access Control to keep those changes separate and provide improved automation user experience for those use cases. In other words, it’s not realistic to have every user perform self-service Terraform changes for access requests.
Access Control uses the Identity Data policies
and manifests
repositories that are focused on who belongs to a group.
The policies
repository is for managing YAML policy files that are managed by humans using the CLI for generating properly formatted changes and has strict merge request approval rules for change management before changes are merged.
The manifests
repository has the data that has been generated by the accessctl
automation, with detected changes separately added in the auditlog
repository for audit preservation.
See the mirrored PoC data repositories for a sneak peak of the future of Identity v3.
graph LR subgraph accessctl GitLab Repositories direction LR subgraph accessctl-policies Repo direction TB REPO_INV_POLICIES_ROLES[("policies/role/{kingdom}.yml")] REPO_INV_POLICIES_OU[("policies/ou/{kingdom}.yml")] end subgraph accessctl-manifests Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")] REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")] REPO_INV_MANIFESTS_ROLE[("manifests/roles/<br/>{role}.yml/json/csv")] REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")] end subgraph accessctl-auditlog Repo direction TB REPO_INV_CHANGELOG_USERS_ONBOARDING[("auditlog/users/<br />onboarding.yml/json/csv")] REPO_INV_CHANGELOG_USERS_OFFBOARDING[("auditlog/users/<br />offboarding.yml/json/csv")] REPO_INV_CHANGELOG_USERS_CHANGES[("auditlog/users/<br />attributes.yml/json/csv")] REPO_INV_CHANGELOG_ATTRIBUTE[("auditlog/attribute/<br />{attribute}.yml/json/csv")] REPO_INV_CHANGELOG_ROLE[("auditlog/role/<br />{role}.yml/json/csv")] REPO_INV_CHANGELOG_OU[("auditlog/ou/<br />{ou}.yml/json/csv")] end end classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px;
Data Security
- Although these could be part of the same repository, we provide separation so that the pipeline automation has read-only access to the policies repository to avoid any potential compromise.
- The SSOT repository is hosted on the Black OPs self-managed GitLab instance to protect the data integrity, and repository data is mirrored to GitLab.com SaaS for team member read-only visibility.
- The runtime pipelines, CI/CD variables, and secrets management are on the Black Ops GitLab instance that no team members have elevated permissions on other than break glass access by the Identity Engineering Team. We use the CLI for generating properly formatted changes via API calls to avoid users making freeform changes or clicking buttons.
Data Flow
graph TB subgraph "Workday" WORKDAY_WORKER[Worker] WORKDAY_REGION([Region]) WORKDAY_DEPARTMENT([Department]) WORKDAY_DIVISION([Division]) WORKDAY_JOBTITLE([Job Title]) WORKDAY_DEPARTMENT -.- WORKDAY_WORKER WORKDAY_DIVISION -.- WORKDAY_WORKER WORKDAY_REGION -.- WORKDAY_WORKER WORKDAY_JOBTITLE -.- WORKDAY_WORKER WORKDAY_WORKER -. "Manager Relationship" .-> WORKDAY_WORKER end subgraph Identity SaaS Vendor Services direction LR subgraph Okta GitLab Tenant OKTA_PROFILE_MAPPING{{Workday to Okta<br>Profile Mapping}} IDENTITY_VENDOR_OKTA_API_USERS[("Okta Users")] IDENTITY_VENDOR_OKTA_API_ENDPOINT[("Okta API<br />https://gitlab.okta.com/api/v1")] WORKDAY_WORKER -- "SCIM Provisioning<br />Every Hour" --> OKTA_PROFILE_MAPPING OKTA_PROFILE_MAPPING --> IDENTITY_VENDOR_OKTA_API_USERS IDENTITY_VENDOR_OKTA_API_USERS --> IDENTITY_VENDOR_OKTA_API_ENDPOINT end end subgraph Automation Plane subgraph Access Control CI/CD Pipeline direction LR CI_STAGE1_MANIFEST["Manifest Stage 1"] CI_STAGE2_CHANGELOG["Auditlog Stage 2"] CI_STAGE3_PROVISIONING["Provisioning Stage 3"] subgraph REST API Client direction LR CI_REST_INTEGRATION["Access Control<br />Integration API"] CI_REST_OKTA["Okta API<br />Groups"] CI_REST_GOOGLE["Google API<br />Group Users"] CI_REST_GITLAB["GitLab API"] end end subgraph Okta API OKTA_GROUPS["Okta rbac_* Groups"]:::emerald OKTA_GROUPS_USERS["Group Users"]:::emerald OKTA_USERS["Okta Users"] OKTA_GROUPS -.- OKTA_GROUPS_USERS OKTA_GROUPS_USERS -.- OKTA_USERS end subgraph Google Workspace Directory API GOOGLE_GROUPS["Google rbac_* Groups"]:::emerald GOOGLE_GROUPS_USERS["Group Members"]:::emerald GOOGLE_USERS["Google Users"] GOOGLE_GROUPS -.- GOOGLE_GROUPS_USERS GOOGLE_GROUPS_USERS -.- GOOGLE_USERS end subgraph GitLab API for SaaS and ops.gitlab.net GITLAB_GROUPS["@gitlab-rbac/* Groups"]:::orange GITLAB_GROUPS_USERS["Group Members"]:::orange GITLAB_USERS["@gitlab.com Users"] GITLAB_GROUPS -.- GITLAB_GROUPS_USERS GITLAB_GROUPS_USERS -.- GITLAB_USERS end end subgraph Control Plane subgraph Slack CTL_SLACK["/accessctl<br />Slack Bot<br />End Users and Approvers"]:::red end subgraph Mac and Linux CTL_CLI["accessctl CLI<br />Power Users and Admins"]:::red end subgraph Web Application CTL_UI["ctl.gitlab.systems<br/>accessctl Identity UI<br />Team Member Self Service"]:::red end end subgraph Data Plane direction TB CTL_API["ctl.gitlab.black<br/>accessctl Identity API<br />(behind Teleport bastion)"]:::re CTL_API -.- REPO_POLICIES CTL_API -.- REPO_MANIFESTS CTL_API -.- REPO_CHANGELOG subgraph Access Control Repositories REPO_POLICIES[("accessctl-policies Repo<br />policies/*.yml")] REPO_MANIFESTS[("accessctl-manifests Repo<br />manifests/*.yml/json/csv")] REPO_CHANGELOG[("accessctl-auditlog Repo<br />auditlog/*.yml/json/csv")] end CTL_S3["Access Control S3 Logs<br />and Repo Backup"] SIEM_DEVO_S3["Devo ingests S3 logs"]:::violet end subgraph Integration Control Plane subgraph Access Control Integration Scripts INTEGRATION_API["Integration API/Webhooks"] INTEGRATION_BLUEPRINT["Blueprints (Future)<br />Script Templates to replace<br />no code use cases for<br />common REST API calls<br/>(Email, Slack Post, Create Issue, <br />Add Row to Sheet, etc)"] INTEGRATION_TRANSACTION["Database Audit Transaction<br />Database rows for each event<br />with metadata for blueprint variables<br />and dispatching jobs on Redis queue."] INTEGRATION_JOB["Queued Jobs"] INTEGRATION_TRANSACTION -.- INTEGRATION_BLUEPRINT INTEGRATION_BLUEPRINT -.- INTEGRATION_JOB INTEGRATION_TRANSACTION --> INTEGRATION_JOB end INTEGRATION_LEGACY_AR_ISSUE["Auto-Generated AR Issue<br/>for legacy audit logging <br />transition in FY25"]:::violet INTEGRATION_GOOGLE_SHEET["Google Sheets<br />Audit Inventory Reports"]:::violet INTEGRATION_SLACK_NOTIFICATION["Slack Notifications<br />to Channel(s) or User(s)"]:::violet end IDENTITY_VENDOR_OKTA_API_ENDPOINT --> CI_STAGE1_MANIFEST CTL_CLI --> CTL_API CTL_SLACK --> CTL_API CTL_UI --> CTL_API CTL_S3 --> SIEM_DEVO_S3 REPO_POLICIES --> CI_STAGE1_MANIFEST CI_STAGE2_CHANGELOG --> REPO_CHANGELOG CI_STAGE2_CHANGELOG --> CTL_S3 CI_STAGE3_PROVISIONING --> CTL_S3 REPO_MANIFESTS --> CI_STAGE3_PROVISIONING CI_STAGE1_MANIFEST --> CI_STAGE2_CHANGELOG CI_STAGE1_MANIFEST --> REPO_MANIFESTS REPO_MANIFESTS --> CI_STAGE2_CHANGELOG CI_STAGE2_CHANGELOG --> CI_STAGE3_PROVISIONING CI_STAGE3_PROVISIONING -----> CI_REST_OKTA CI_STAGE3_PROVISIONING --> CI_REST_GOOGLE CI_STAGE3_PROVISIONING --> CI_REST_GITLAB CI_STAGE2_CHANGELOG ----> CI_REST_INTEGRATION CI_STAGE3_PROVISIONING ----> CI_REST_INTEGRATION CI_REST_INTEGRATION ------> INTEGRATION_API INTEGRATION_API --> INTEGRATION_JOB INTEGRATION_JOB --> INTEGRATION_LEGACY_AR_ISSUE INTEGRATION_JOB --> INTEGRATION_GOOGLE_SHEET INTEGRATION_JOB --> INTEGRATION_SLACK_NOTIFICATION CI_REST_OKTA --> OKTA_GROUPS_USERS CI_REST_OKTA --> OKTA_GROUPS CI_REST_GOOGLE --> GOOGLE_GROUPS_USERS CI_REST_GOOGLE --> GOOGLE_GROUPS CI_REST_GITLAB --> GITLAB_GROUPS CI_REST_GITLAB --> GITLAB_GROUPS_USERS classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px;
Access Control (accessctl)
Access Control is an open source monolith code base created by Jeff Martin that is built with the Laravel framework with multiple functional components included that are deployed in a distributed fashion to dogfood GitLab CI/CD pipelines and GitLab Runner for background jobs, GitLab repository for policy and manifest storage, CODEOWNERS and merge request approvals for change management.
Although Laravel has batteries included for all of the full stack web application and SQL database capabilities that rivals Django or Ruby on Rails, we are using it as more of a scripting engine that is easier for team members to contribute to and leverages GitLab for backend processing wherever possible. We still use Laravel native capabilities to expose a REST API, Web UI, and CLI for improved frontend user experience.
We use Laravel Actions for our business logic service classes. By using Laravel Actions, we simplify our architecture since each single responsibility principle (SRP) action can be invoked as a class, a console CLI command, or a background job. This also allows us to get data from the GitLab API for repository data instead of needing to use models and migrations for a large SQL database.
See the gitlab-identity/accessctl repository and README to learn more.
Why Laravel
It’s a simple case of using the tools that we know and can deliver business value efficiently. Jeff Martin is the primary developer has been using it for 10+ years and has tried using Ruby on Rails, Golang, Python, and still finds that the Laravel ecosystem is richer.
You may have already seen what’s possible with GitLab Sandbox Cloud, Demo Systems, and Training Lab Manager. It just works and only requires 1 engineer part time to take us to production. We’ve taught more than a dozen team members how Laravel works without any trouble. This isn’t part of the GitLab product itself so that’s a non factor. Please reach out to Jeff for a code pair if you’re interested.
CI/CD Scripts
We use a Docker image with the accessctl
application with GitLab Runner to run Artisan Console commands that from Laravel actions. Each CI/CD job is usually related to ETL operations for (E)xtracting IAM/RBAC data from vendor APIs, (T)ransforming the API responses into standardized YAML and JSON array schemas, and (L)oading the data into the GitLab repositories (accessctl-manifests
, accessctl-auditlog
and accessctl-policies
for system automation) and Google Sheets (for human use).
The CI/CD pipeline has several stages:
API
We have the accessctl
application deployed at https://ctl.gitlab.black
that provides a REST API for data stored in the GitLab repository that is proxied and formatted as if we used a traditional SQL database. This improves the consumability of our data, and provides an endpoint for CLI application.
This is deployed separately from the Web UI since we use network ACLs to only allow access for users connected to the VPN or service accounts that have pre-configured allowlist IP addresses.
Web UI
We have the accessctl
application deployed at https://ctl.gitlab.systems
that provides Web UI with Okta authentication (with 2FA and Okta Verify device trust) for end user team members and managers to perform day-to-day tasks without administrative access.
For security and UI design internal tool efficiency reasons, only the CLI is used for admin actions.
Slack Bot
We will be building a Slack bot that will be used for end user form wizard workflows and just-in-time/manager approvals to supplement the Web UI.
CLI
The accessctl
CLI is used by power users and admins for generating properly formatted MRs with policy changes with the accessctl API.
We only accept system generated changes. No human generated (manual edit) MRs are allowed.
Access Check (accesschk)
Access Check is a series of scripts for performing audit evidence collection. It is part of the accessctl codebase but is run using an independent CI/CD pipeline.
You can think of this as an audit automation platform that also provides a single source of truth (SSOT) data source for the latest aggregated information available from each vendor’s API and saving it as a CSV/JSON/YML file and Google Sheet that can be referenced by auditors, compliance teams, and other team members as needed.
See the accesschk handbook page to learn more.
Additional Reading
See the Identity Platform Architecture and User Guide links at the top of this page.
Identity Platform CI/CD Auditlog Pipeline
Not Live Yet
You are viewing a preview of documentation for the future state of GitLab Identity v3 (mid 2024). See the Access Management Policy for the GitLab Identity v2 current state with baseline entitlements and access requests. See the roadmap in the epics gantt chart.Pipeline Overview
graph LR subgraph accessctl GitLab Repositories direction LR subgraph accessctl-policies Repo direction TB REPO_INV_POLICIES_ROLES[("policies/role/{kingdom}.yml")] REPO_INV_POLICIES_OU[("policies/ou/{kingdom}.yml")] end subgraph accessctl-manifests Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")]:::sky REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")]:::sky REPO_INV_MANIFESTS_ROLE[("manifests/roles/<br/>{role}.yml/json/csv")]:::sky REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")]:::sky end subgraph accessctl-auditlog Repo direction TB REPO_INV_CHANGELOG_USERS_ONBOARDING[("auditlog/users/<br />onboarding.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_USERS_OFFBOARDING[("auditlog/users/<br />offboarding.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_USERS_CHANGES[("auditlog/users/<br />attributes.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_ATTRIBUTE[("auditlog/attribute/<br />{attribute}.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_ROLE[("auditlog/role/<br />{role}.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_OU[("auditlog/ou/<br />{ou}.yml/json/csv")]:::emerald end end subgraph accessctl GitLab CI/CD Pipeline Jobs direction LR subgraph Manifest Stage direction LR CI_MANIFEST_USER_JOB[["Stage 1.1<br />Users Job<br/>CLI manifest:users"]] CI_MANIFEST_ROLE_JOB[["Stage 1.2<br />Roles Job<br />CLI manifest:roles"]] CI_MANIFEST_GROUP_JOB[["Stage 1.3<br />Org Units Job<br/>CLI manifest:ou"]] end subgraph Auditlog Stage direction LR CI_CHANGELOG_USER_JOB[["Stage 2.1<br />Users Job<br/>CLI auditlog:users"]]:::orange CI_CHANGELOG_ATTRIBUTE_JOB[["Stage 2.2<br />Attributes Job<br/>CLI auditlog:attributes"]]:::orange CI_CHANGELOG_ROLE_JOB[["Stage 2.3<br />Roles Job<br/>CLI auditlog:roles"]]:::orange CI_CHANGELOG_GROUP_JOB[["Stage 2.4<br />Org Units Job<br/>CLI auditlog:ou"]]:::orange end subgraph Provisioning Stage direction LR subgraph Okta API CI_PROVISIONING_OKTA_USER_JOB["Stage 3.1<br />Okta Users Job<br />provision:okta-users"] CI_PROVISIONING_OKTA_GROUP_JOB["Stage 3.2<br />Okta Groups Job<br />provision:okta-groups"] end subgraph Google Workspace Directory API CI_PROVISIONING_GOOGLE_GROUP_JOB["Stage 3.3<br />Google Groups Job<br />provision:google-groups"] end subgraph GitLab.com SaaS API CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB["Stage 3.4<br />GitLab SaaS Groups Job<br />provision:gitlab-saas-groups"] end subgraph GitLab Self-Managed Instance API CI_PROVISIONING_GITLAB_SELF_GROUP_JOB["Stage 3.5<br />GitLab Self-Managed Groups Job<br />provision:gitlab-self-groups"] end end CI_MANIFEST_USER_JOB --> CI_MANIFEST_ROLE_JOB --> CI_MANIFEST_GROUP_JOB CI_CHANGELOG_USER_JOB --> CI_CHANGELOG_ATTRIBUTE_JOB --> CI_CHANGELOG_ROLE_JOB --> CI_CHANGELOG_GROUP_JOB CI_PROVISIONING_OKTA_USER_JOB --> CI_PROVISIONING_OKTA_GROUP_JOB --> CI_PROVISIONING_GOOGLE_GROUP_JOB --> CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB --> CI_PROVISIONING_GITLAB_SELF_GROUP_JOB click CI_MANIFEST_USER_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_MANIFEST_ROLE_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_MANIFEST_GROUP_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_CHANGELOG_USER_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_ATTRIBUTE_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_ROLE_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_GROUP_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_PROVISIONING_OKTA_USER_JOB "/handbook/security/identity/platform/provisioning/okta" "View Details" click CI_PROVISIONING_OKTA_GROUP_JOB "/handbook/security/identity/platform/provisioning/okta" "View Details" click CI_PROVISIONING_GOOGLE_GROUP_JOB "/handbook/security/identity/platform/provisioning/google" "View Details" click CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB "/handbook/security/identity/platform/provisioning/gitlab" "View Details" click CI_PROVISIONING_GITLAB_SELF_GROUP_JOB "/handbook/security/identity/platform/provisioning/gitlab" "View Details" classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px; end
CI/CD Job Workflows
Users Auditlog
graph TB subgraph Identity GitLab Repositories subgraph accessctl-manifests Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")]:::sky REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")] REPO_INV_MANIFESTS_ROLES[("manifests/roles/<br />{role}.yml/json/csv")] REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")] end end subgraph Identity Platform CI/CD Auditlog Stage Scripts direction LR CI_USER_JOB[["Stage 2.1<br />Users Job<br/>CLI auditlog:users"]]:::orange CI_USER_PARSE_MANIFEST_CURRENT[(Parse Current Manifest Commit)] CI_USER_PARSE_MANIFEST_PREVIOUS[(Parse Previous Manifest Commit)] CI_USER_PARSE_MANIFEST_FILTER{Filter results<br />with differences} CI_USER_ONBOARDING_CHECK{{Check if user has been created}} CI_USER_OFFBOARDING_CHECK{{Check if user has been deprovisioned}} CI_USER_ATTRIBUTE_CHECK{{Loop through users and<br />compare attributes for differences}} CI_USER_LOG_S3{{Create Audit Log entry in S3 bucket}} CI_USER_API{{Audit Transaction<br />REST API Call to accessctl<br />for automation workflows}} CI_USER_LOG_REPO{{Append Auditlog in Repository}} CI_USER_JOB --> CI_USER_PARSE_MANIFEST_CURRENT CI_USER_JOB --> CI_USER_PARSE_MANIFEST_PREVIOUS CI_USER_PARSE_MANIFEST_CURRENT --> CI_USER_PARSE_MANIFEST_FILTER CI_USER_PARSE_MANIFEST_PREVIOUS --> CI_USER_PARSE_MANIFEST_FILTER CI_USER_PARSE_MANIFEST_FILTER --> CI_USER_ONBOARDING_CHECK CI_USER_PARSE_MANIFEST_FILTER --> CI_USER_OFFBOARDING_CHECK CI_USER_PARSE_MANIFEST_FILTER --> CI_USER_ATTRIBUTE_CHECK CI_USER_ONBOARDING_CHECK --> CI_USER_LOG_S3 CI_USER_OFFBOARDING_CHECK --> CI_USER_LOG_S3 CI_USER_ATTRIBUTE_CHECK --> CI_USER_LOG_S3 CI_USER_LOG_S3 --> CI_USER_API CI_USER_API --> CI_USER_LOG_REPO end subgraph Identity GitLab Repositories subgraph accessctl-auditlog Repo direction TB REPO_INV_CHANGELOG_USERS_ONBOARDING[("auditlog/users/<br />onboarding.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_USERS_OFFBOARDING[("auditlog/users/<br />offboarding.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_USERS_CHANGES[("auditlog/users/<br />attributes.yml/json/csv")]:::emerald end end REPO_INV_MANIFESTS_USERS --> CI_USER_JOB CI_USER_LOG_REPO --> REPO_INV_CHANGELOG_USERS_ONBOARDING CI_USER_LOG_REPO --> REPO_INV_CHANGELOG_USERS_OFFBOARDING CI_USER_LOG_REPO --> REPO_INV_CHANGELOG_USERS_CHANGES classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px;
Attributes Auditlog
graph TB subgraph Identity GitLab Repositories subgraph accessctl-manifests Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")] REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")]:::sky REPO_INV_MANIFESTS_ROLES[("manifests/roles/<br />{role}.yml/json/csv")] REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")] end end subgraph Identity Platform CI/CD Auditlog Stage Scripts direction LR CI_ATTRIBUTE_JOB[["Stage 2.2<br />Attributes Job<br/>CLI auditlog:attributes"]]:::orange CI_ATTRIBUTE_PARSE_MANIFEST_CURRENT[(Parse Current Manifest Commit)] CI_ATTRIBUTE_PARSE_MANIFEST_PREVIOUS[(Parse Previous Manifest Commit)] CI_ATTRIBUTE_PARSE_MANIFEST_FILTER{Filter results<br />with differences} CI_ATTRIBUTE_CREATED_CHECK{{Check if attribute has been created}} CI_ATTRIBUTE_DELETED_CHECK{{Check if attribute has been deleted}} CI_ATTRIBUTE_LOG_S3{{Create Audit Log entry in S3 bucket}} CI_ATTRIBUTE_API{{Audit Transaction<br />REST API Call to accessctl<br />for automation workflows}} CI_ATTRIBUTE_LOG_REPO{{Append Auditlog in Repository}} CI_ATTRIBUTE_JOB --> CI_ATTRIBUTE_PARSE_MANIFEST_CURRENT CI_ATTRIBUTE_JOB --> CI_ATTRIBUTE_PARSE_MANIFEST_PREVIOUS CI_ATTRIBUTE_PARSE_MANIFEST_CURRENT --> CI_ATTRIBUTE_PARSE_MANIFEST_FILTER CI_ATTRIBUTE_PARSE_MANIFEST_PREVIOUS --> CI_ATTRIBUTE_PARSE_MANIFEST_FILTER CI_ATTRIBUTE_PARSE_MANIFEST_FILTER --> CI_ATTRIBUTE_CREATED_CHECK CI_ATTRIBUTE_PARSE_MANIFEST_FILTER --> CI_ATTRIBUTE_DELETED_CHECK CI_ATTRIBUTE_CREATED_CHECK --> CI_ATTRIBUTE_LOG_S3 CI_ATTRIBUTE_DELETED_CHECK --> CI_ATTRIBUTE_LOG_S3 CI_ATTRIBUTE_LOG_S3 --> CI_ATTRIBUTE_API CI_ATTRIBUTE_API --> CI_ATTRIBUTE_LOG_REPO end subgraph Identity GitLab Repositories subgraph accessctl-auditlog Repo direction TB REPO_INV_CHANGELOG_ATTRIBUTE[("auditlog/attribute/<br />{attribute}.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_ROLE[("auditlog/role/<br />{role}.yml/json/csv")] REPO_INV_CHANGELOG_OU[("auditlog/ou/<br />{ou}.yml/json/csv")] end end REPO_INV_MANIFESTS_ATTRIBUTES --> CI_ATTRIBUTE_JOB CI_ATTRIBUTE_LOG_REPO --> REPO_INV_CHANGELOG_ATTRIBUTE classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px;
Roles Auditlog
graph TB subgraph Identity GitLab Repositories subgraph accessctl-manifests Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")] REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")] REPO_INV_MANIFESTS_ROLES[("manifests/roles/<br />{role}.yml/json/csv")]:::sky REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")] end end subgraph Identity Platform CI/CD Auditlog Stage Scripts direction LR CI_ROLE_JOB[["Stage 2.3<br />Roles Job<br/>CLI auditlog:roles"]]:::orange CI_ROLE_PARSE_MANIFEST_CURRENT[(Parse Current Manifest Commit)] CI_ROLE_PARSE_MANIFEST_PREVIOUS[(Parse Previous Manifest Commit)] CI_ROLE_PARSE_MANIFEST_FILTER{Filter results<br />with differences} CI_ROLE_CREATED_CHECK{{Check if user has been added}} CI_ROLE_DELETED_CHECK{{Check if user has been deleted}} CI_ROLE_LOG_S3{{Create Audit Log entry in S3 bucket}} CI_ROLE_API{{Audit Transaction<br />REST API Call to accessctl<br />for automation workflows}} CI_ROLE_LOG_REPO{{Append Auditlog in Repository}} CI_ROLE_JOB --> CI_ROLE_PARSE_MANIFEST_CURRENT CI_ROLE_JOB --> CI_ROLE_PARSE_MANIFEST_PREVIOUS CI_ROLE_PARSE_MANIFEST_CURRENT --> CI_ROLE_PARSE_MANIFEST_FILTER CI_ROLE_PARSE_MANIFEST_PREVIOUS --> CI_ROLE_PARSE_MANIFEST_FILTER CI_ROLE_PARSE_MANIFEST_FILTER --> CI_ROLE_CREATED_CHECK CI_ROLE_PARSE_MANIFEST_FILTER --> CI_ROLE_DELETED_CHECK CI_ROLE_CREATED_CHECK --> CI_ROLE_LOG_S3 CI_ROLE_DELETED_CHECK --> CI_ROLE_LOG_S3 CI_ROLE_LOG_S3 --> CI_ROLE_API CI_ROLE_API --> CI_ROLE_LOG_REPO end subgraph Identity GitLab Repositories subgraph accessctl-auditlog Repo direction TB REPO_INV_CHANGELOG_ATTRIBUTE[("auditlog/attribute/<br />{attribute}.yml/json/csv")] REPO_INV_CHANGELOG_ROLE[("auditlog/role/<br />{role}.yml/json/csv")]:::emerald REPO_INV_CHANGELOG_OU[("auditlog/ou/<br />{ou}.yml/json/csv")] end end REPO_INV_MANIFESTS_ROLES --> CI_ROLE_JOB CI_ROLE_LOG_REPO --> REPO_INV_CHANGELOG_ROLE classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px;
Organization Units Auditlog
graph TB subgraph Identity GitLab Repositories subgraph accessctl-manifests Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")] REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")] REPO_INV_MANIFESTS_ROLES[("manifests/roles/<br />{role}.yml/json/csv")] REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")]:::sky end end subgraph Identity Platform CI/CD Auditlog Stage Scripts direction LR CI_GROUP_JOB[["Stage 2.4<br />Org Units Job<br/>CLI auditlog:ou"]]:::orange CI_GROUP_PARSE_MANIFEST_CURRENT[(Parse Current Manifest Commit)] CI_GROUP_PARSE_MANIFEST_PREVIOUS[(Parse Previous Manifest Commit)] CI_GROUP_PARSE_MANIFEST_FILTER{Filter results<br />with differences} CI_GROUP_CREATED_CHECK{{Check if user has been added}} CI_GROUP_DELETED_CHECK{{Check if user has been deleted}} CI_GROUP_LOG_S3{{Create Audit Log entry in S3 bucket}} CI_GROUP_API{{Audit Transaction<br />REST API Call to accessctl<br />for automation workflows}} CI_GROUP_LOG_REPO{{Append Auditlog in Repository}} CI_GROUP_JOB --> CI_GROUP_PARSE_MANIFEST_CURRENT CI_GROUP_JOB --> CI_GROUP_PARSE_MANIFEST_PREVIOUS CI_GROUP_PARSE_MANIFEST_CURRENT --> CI_GROUP_PARSE_MANIFEST_FILTER CI_GROUP_PARSE_MANIFEST_PREVIOUS --> CI_GROUP_PARSE_MANIFEST_FILTER CI_GROUP_PARSE_MANIFEST_FILTER --> CI_GROUP_CREATED_CHECK CI_GROUP_PARSE_MANIFEST_FILTER --> CI_GROUP_DELETED_CHECK CI_GROUP_CREATED_CHECK --> CI_GROUP_LOG_S3 CI_GROUP_DELETED_CHECK --> CI_GROUP_LOG_S3 CI_GROUP_LOG_S3 --> CI_GROUP_API CI_GROUP_API --> CI_GROUP_LOG_REPO end subgraph Identity GitLab Repositories subgraph accessctl-auditlog Repo direction TB REPO_INV_CHANGELOG_ATTRIBUTE[("auditlog/attribute/<br />{attribute}.yml/json/csv")] REPO_INV_CHANGELOG_ROLE[("auditlog/role/<br />{role}.yml/json/csv")] REPO_INV_CHANGELOG_OU[("auditlog/ou/<br />{ou}.yml/json/csv")]:::emerald end end REPO_INV_MANIFESTS_OU --> CI_GROUP_JOB CI_GROUP_LOG_REPO --> REPO_INV_CHANGELOG_OU classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px;
Identity Platform CI/CD Manifest Pipeline
Not Live Yet
You are viewing a preview of documentation for the future state of GitLab Identity v3 (mid 2024). See the Access Management Policy for the GitLab Identity v2 current state with baseline entitlements and access requests. See the roadmap in the epics gantt chart.Pipeline Overview
graph LR subgraph accessctl GitLab Repositories direction LR subgraph accessctl-policies Repo direction TB REPO_INV_POLICIES_ROLES[("policies/role/{kingdom}.yml")]:::sky REPO_INV_POLICIES_OU[("policies/ou/{kingdom}.yml")]:::sky end subgraph accessctl-manifests Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")]:::emerald REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")]:::emerald REPO_INV_MANIFESTS_ROLE[("manifests/roles/<br/>{role}.yml/json/csv")]:::emerald REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")]:::emerald end subgraph accessctl-auditlog Repo direction TB REPO_INV_CHANGELOG_USERS_ONBOARDING[("auditlog/users/<br />onboarding.yml/json/csv")] REPO_INV_CHANGELOG_USERS_OFFBOARDING[("auditlog/users/<br />offboarding.yml/json/csv")] REPO_INV_CHANGELOG_USERS_CHANGES[("auditlog/users/<br />attributes.yml/json/csv")] REPO_INV_CHANGELOG_ATTRIBUTE[("auditlog/attribute/<br />{attribute}.yml/json/csv")] REPO_INV_CHANGELOG_ROLE[("auditlog/role/<br />{role}.yml/json/csv")] REPO_INV_CHANGELOG_OU[("auditlog/ou/<br />{ou}.yml/json/csv")] end end subgraph accessctl GitLab CI/CD Pipeline Jobs direction LR subgraph Manifest Stage direction LR CI_MANIFEST_USER_JOB[["Stage 1.1<br />Users Job<br/>CLI manifest:users"]]:::orange CI_MANIFEST_ROLE_JOB[["Stage 1.2<br />Roles Job<br />CLI manifest:roles"]]:::orange CI_MANIFEST_GROUP_JOB[["Stage 1.3<br />Org Units Job<br/>CLI manifest:ou"]]:::orange end subgraph Auditlog Stage direction LR CI_CHANGELOG_USER_JOB[["Stage 2.1<br />Users Job<br/>CLI auditlog:users"]] CI_CHANGELOG_ATTRIBUTE_JOB[["Stage 2.2<br />Attributes Job<br/>CLI auditlog:attributes"]] CI_CHANGELOG_ROLE_JOB[["Stage 2.3<br />Roles Job<br/>CLI auditlog:roles"]] CI_CHANGELOG_GROUP_JOB[["Stage 2.4<br />Org Units Job<br/>CLI auditlog:ou"]] end subgraph Provisioning Stage direction LR subgraph Okta API CI_PROVISIONING_OKTA_USER_JOB["Stage 3.1<br />Okta Users Job<br />provision:okta-users"] CI_PROVISIONING_OKTA_GROUP_JOB["Stage 3.2<br />Okta Groups Job<br />provision:okta-groups"] end subgraph Google Workspace Directory API CI_PROVISIONING_GOOGLE_GROUP_JOB["Stage 3.3<br />Google Groups Job<br />provision:google-groups"] end subgraph GitLab.com SaaS API CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB["Stage 3.4<br />GitLab SaaS Groups Job<br />provision:gitlab-saas-groups"] end subgraph GitLab Self-Managed Instance API CI_PROVISIONING_GITLAB_SELF_GROUP_JOB["Stage 3.5<br />GitLab Self-Managed Groups Job<br />provision:gitlab-self-groups"] end end CI_MANIFEST_USER_JOB --> CI_MANIFEST_ROLE_JOB --> CI_MANIFEST_GROUP_JOB CI_CHANGELOG_USER_JOB --> CI_CHANGELOG_ATTRIBUTE_JOB --> CI_CHANGELOG_ROLE_JOB --> CI_CHANGELOG_GROUP_JOB CI_PROVISIONING_OKTA_USER_JOB --> CI_PROVISIONING_OKTA_GROUP_JOB --> CI_PROVISIONING_GOOGLE_GROUP_JOB --> CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB --> CI_PROVISIONING_GITLAB_SELF_GROUP_JOB click CI_MANIFEST_USER_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_MANIFEST_ROLE_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_MANIFEST_GROUP_JOB "/handbook/security/identity/platform/manifests" "View Details" click CI_CHANGELOG_USER_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_ATTRIBUTE_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_ROLE_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_CHANGELOG_GROUP_JOB "/handbook/security/identity/platform/auditlog" "View Details" click CI_PROVISIONING_OKTA_USER_JOB "/handbook/security/identity/platform/provisioning/okta" "View Details" click CI_PROVISIONING_OKTA_GROUP_JOB "/handbook/security/identity/platform/provisioning/okta" "View Details" click CI_PROVISIONING_GOOGLE_GROUP_JOB "/handbook/security/identity/platform/provisioning/google" "View Details" click CI_PROVISIONING_GITLAB_SAAS_GROUP_JOB "/handbook/security/identity/platform/provisioning/gitlab" "View Details" click CI_PROVISIONING_GITLAB_SELF_GROUP_JOB "/handbook/security/identity/platform/provisioning/gitlab" "View Details" classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px; end
CI/CD Job Workflows
graph TB subgraph "Workday" WORKDAY_WORKER[Worker] WORKDAY_REGION([Region]) WORKDAY_DEPARTMENT([Department]) WORKDAY_DIVISION([Division]) WORKDAY_JOBTITLE([Job Title]) WORKDAY_DEPARTMENT -.- WORKDAY_WORKER WORKDAY_DIVISION -.- WORKDAY_WORKER WORKDAY_REGION -.- WORKDAY_WORKER WORKDAY_JOBTITLE -.- WORKDAY_WORKER WORKDAY_WORKER -. "Manager Relationship" .-> WORKDAY_WORKER end subgraph Identity SaaS Vendor Services direction LR subgraph Okta GitLab Tenant OKTA_PROFILE_MAPPING{{Workday to Okta<br>Profile Mapping}} IDENTITY_VENDOR_OKTA_API_USERS[("Okta Users")] IDENTITY_VENDOR_OKTA_API_ENDPOINT[("Okta API<br />https://gitlab.okta.com/api/v1")] WORKDAY_WORKER -- "SCIM Provisioning<br />Every Hour" --> OKTA_PROFILE_MAPPING OKTA_PROFILE_MAPPING --> IDENTITY_VENDOR_OKTA_API_USERS IDENTITY_VENDOR_OKTA_API_USERS --> IDENTITY_VENDOR_OKTA_API_ENDPOINT end end subgraph Identity Platform CI/CD Manifest Stage Scripts direction LR CI_USER_JOB[["Stage 1.1<br />Users Job<br/>CLI manifest:users"]]:::orange CI_ROLE_JOB[["Stage 1.2<br />Roles Job<br />CLI manifest:roles"]]:::orange CI_GROUP_JOB[["Stage 1.3<br />Org Units Job<br/>CLI manifest:ou"]]:::orange CI_USER_PARSE_API[(Get All Users<br/>from Okta API)] CI_USER_TRANSFORM{{Transform users with<br/>snake_case attributes}} CI_USER_MANIFEST_USERS{{Create manifest of users}} CI_USER_MANIFEST_ATTRIBUTES{{Create manifest of unique<br />attributes and count of users}} CI_USER_JOB --> CI_USER_PARSE_API CI_USER_PARSE_API --> CI_USER_TRANSFORM CI_USER_TRANSFORM --> CI_USER_MANIFEST_USERS CI_USER_MANIFEST_USERS ----> CI_USER_MANIFEST_ATTRIBUTES CI_ROLE_PARSE_POLICY[(Parse Users<br />Parse Attributes<br />Parse Policies<br />Parse Policy Rules)] CI_ROLE_VALIDATE_KEYS{{Validate attribute keys exist<br />for each policy rule}} CI_ROLE_GET_NAMED_USERS{{Get users<br />for each policy handle}} CI_ROLE_GET_USERS_FOR_ATTRIBUTE{{Get users<br />for each Attribute Key}} CI_ROLE_CREATE_MANIFEST{{Create manifest of users<br />with unique email addresses<br />for each role}} CI_ROLE_JOB --> CI_ROLE_PARSE_POLICY CI_ROLE_PARSE_POLICY --> CI_ROLE_VALIDATE_KEYS CI_ROLE_VALIDATE_KEYS --> CI_ROLE_GET_NAMED_USERS CI_ROLE_GET_NAMED_USERS --> CI_ROLE_GET_USERS_FOR_ATTRIBUTE CI_ROLE_GET_USERS_FOR_ATTRIBUTE ---> CI_ROLE_CREATE_MANIFEST CI_GROUP_PARSE_POLICY[(Parse Users<br />Parse Roles<br />Parse Attributes<br />Parse Policies<br />Parse Policy Rules)] CI_GROUP_VALIDATE_KEYS{{Validate attribute keys<br /> and role keys exist<br />for each policy rule}} CI_GROUP_GET_NAMED_USERS{{Get users<br />for each policy handle}} CI_GROUP_GET_USERS_FOR_ROLE{{Get users<br />for each Role Key}} CI_GROUP_GET_USERS_FOR_DIMENSION{{Get users<br />for each Attribute Key}} CI_GROUP_CREATE_MANIFEST{{Create manifest<br />with unique email addresses<br />for each group}} CI_GROUP_JOB --> CI_GROUP_PARSE_POLICY CI_GROUP_PARSE_POLICY --> CI_GROUP_VALIDATE_KEYS CI_GROUP_VALIDATE_KEYS --> CI_GROUP_GET_NAMED_USERS CI_GROUP_GET_NAMED_USERS --> CI_GROUP_GET_USERS_FOR_ROLE CI_GROUP_GET_USERS_FOR_ROLE --> CI_GROUP_GET_USERS_FOR_DIMENSION CI_GROUP_GET_USERS_FOR_DIMENSION --> CI_GROUP_CREATE_MANIFEST end subgraph Identity GitLab Repositories subgraph accessctl-policies Repo direction LR REPO_INV_POLICIES_ROLES[("policies/role/{kingdom}.yml")]:::sky REPO_INV_POLICIES_OU[("policies/ou/{kingdom}.yml")]:::sky end end subgraph Identity GitLab Repositories subgraph accessctl-inventory Repo direction TB REPO_INV_MANIFESTS_USERS[("manifests/users/<br />users.yml/json/csv")]:::emerald REPO_INV_MANIFESTS_ATTRIBUTES[("manifests/attributes/<br />{attribute}.yml/json/csv")]:::emerald REPO_INV_MANIFESTS_ROLE[("manifests/roles/<br/>{role}.yml/json/csv")]:::emerald REPO_INV_MANIFESTS_OU[("manifests/ou/<br />{ou}.yml/json/csv")]:::emerald end end CI_USER_MANIFEST_USERS --> REPO_INV_MANIFESTS_USERS CI_USER_MANIFEST_ATTRIBUTES --> REPO_INV_MANIFESTS_ATTRIBUTES IDENTITY_VENDOR_OKTA_API_ENDPOINT --> CI_USER_JOB REPO_INV_POLICIES_ROLES --> CI_ROLE_JOB REPO_INV_POLICIES_OU --> CI_GROUP_JOB CI_ROLE_CREATE_MANIFEST --> REPO_INV_MANIFESTS_ROLE CI_GROUP_CREATE_MANIFEST --> REPO_INV_MANIFESTS_GROUPS classDef slate fill:#cbd5e1,stroke:#475569,stroke-width:1px; classDef red fill:#fca5a5,stroke:#dc2626,stroke-width:1px; classDef orange fill:#fdba74,stroke:#ea580c,stroke-width:1px; classDef yellow fill:#fcd34d,stroke:#ca8a04,stroke-width:1px; classDef emerald fill:#6ee7b7,stroke:#059669,stroke-width:1px; classDef cyan fill:#67e8f9,stroke:#0891b2,stroke-width:1px; classDef sky fill:#7dd3fc,stroke:#0284c7,stroke-width:1px; classDef violet fill:#c4b5fd,stroke:#7c3aed,stroke-width:1px; classDef fuchsia fill:#f0abfc,stroke:#c026d3,stroke-width:1px;
Policy and Manifest Breaking Changes
Attribute Key No Longer Exists
When a policy is parsed prior to creating a manifest, the manifests/attributes/{attribute}.yml
file is parsed to verify that the attribute key exists. This file includes the latest list of unique values from users in the Okta API.
Identity Platform CI/CD Provisioning Pipeline
Work in Progress
This page is a work-in-progress. Please check back later for up-to-date details.See the specific pages for each of our API clients.
55741fb9
)