Skip to main content
Version: Latest

GitLab

Proxy for the GitLab REST API via the managed gateway.

What this connector does

Gives agents gated access to a GitLab instance (SaaS or self-hosted) for:

  • Listing users, projects, merge requests.
  • Creating and editing merge requests.
  • Commenting on merge requests.
  • Minting project access tokens.
  • Listing container-registry repositories and tags.

All operations are gated by PBAC policy. Every call flows: agent → gateway → /introspect (with this connector's resource_type) → OPA decision → GitLab API → response.

The connector supports two authentication modes, chosen at install time:

  • Per-user passthrough (idp_passthrough, default) — the gateway forwards each calling user's own GitLab OAuth token (captured when they log in to PolicyArc through the GitLab IdP). GitLab enforces its own project membership and permissions, and its audit trail shows the real person.
  • Shared service token (static) — every call uses one operator-held personal/project/group access token via the PRIVATE-TOKEN header. PolicyArc still enforces per-user policy and audit in front, but GitLab sees a single service identity.

Prerequisites

  • The base URL of your GitLab instance (defaults to https://gitlab.com).
  • For passthrough mode (default):
    • A GitLab OAuth application registered on your GitLab instance
    • A GitLab IdP registered in PolicyArc with provider key gitlab (use the GitLab preset in the IdP setup wizard)
    • The OAuth application must request the api scope in addition to the OIDC login scopes (openid profile email)
    • Users must log in to PolicyArc through that GitLab IdP — the gateway replays the token captured at login
  • For static mode:
    • A GitLab access token that has enough permission for the operations you want to gate. Options:
      • Personal Access Token — scoped to one user; simplest for PoC.
      • Project Access Token — scoped to one project; recommended for production if the agent only needs one project.
      • Group Access Token — scoped to a group of projects.

Install

1. Set up authentication

For passthrough mode (default)

  1. Create a GitLab OAuth application:

    • On your GitLab instance, go to User Settings → Applications (or Admin Area → Applications for instance-wide)
    • Name: Choose any name (e.g., "PolicyArc MR-75")
    • Redirect URI: Your PolicyArc tenant's callback URL: https://<your-tenant>/oauth2/callback
    • Confidential: Yes (check the box)
    • Scopes: Select openid, profile, email, and api
    • Click Save application and note the Application ID and Secret
  2. Register the GitLab IdP in PolicyArc:

    • In your tenant's admin UI: Identity Providers → Add → GitLab preset
    • Issuer: Your GitLab URL (e.g., https://git.identos.ca)
    • Client ID: The Application ID from step 1
    • Client Secret: The Secret from step 1
    • The preset auto-discovers the OIDC endpoints

No credential provisioning needed — users log in with their own GitLab accounts.

For static mode

Create a GitLab access token with the minimum scopes you need. For MR automation:

OperationMinimum GitLab scope
List projects/MRs/notesread_api
Create/edit MRs, post commentsapi
Create project access tokensapi + project Owner/Maintainer
Container registryread_registry

Put the token in an environment variable the PBAC AS can read:

export GITLAB_PRIVATE_TOKEN="glpat-..."

2. Install the connector

From the admin UI:

  1. Resources → Add → GitLab
  2. Choose authentication mode:
    • Authentication: Select idp_passthrough (default) or static
  3. Fill in the fields:
    • GitLab base URLhttps://gitlab.com or your self-hosted URL
    • Access token (env var name) — (static mode only) the name of the env var holding the token (e.g. GITLAB_PRIVATE_TOKEN), not the token itself
  4. Install

The admin UI reads setup_fields directly from this connector's manifest — same fields shown below in the Setup fields reference.

3. Verify

Request a token with a GitLab scope:

curl -X POST https://<your-as>/token \
-u "<agent-client-id>:<agent-client-secret>" \
-d "grant_type=client_credentials&scope=gitlab:read"

Call through the gateway:

curl https://<your-as>/gateway/identos.gitlab/api/v4/projects?search=acme \
-H "Authorization: Bearer <token>"

A successful 200 with the project list confirms: token issued → introspect allowed → gateway proxied → upstream responded.

What can go wrong

SymptomCauseFix
401 from GitLab (static)GITLAB_PRIVATE_TOKEN env var not set or wrongCheck the AS process environment; confirm echo $GITLAB_PRIVATE_TOKEN resolves in the container.
401 from GitLab (passthrough)User's GitLab token expired or invalidUser needs to re-authenticate through the GitLab IdP.
403 from the gatewayAgent lacks the required scopeRequest a token with gitlab:read, gitlab:write, or gitlab:admin matching the route.
404 at the gatewayWrong pathGateway path is /gateway/identos.gitlab/<GitLab REST path> — GitLab paths start with /api/v4/....
502 "no stored IdP token" (passthrough)The caller never logged in through the gitlab IdPUse authorization_code flow to authenticate via GitLab, not client_credentials.
Agent sees an empty project listThe user has no projects (passthrough) or PAT user has no projects (static)Check GitLab permissions.

Scopes

ScopeMeaningRoutes
gitlab:readRead-only operationslist/get projects, MRs, notes, registries
gitlab:writeMutating operationscreate/edit MRs, post MR comments
gitlab:adminPrivileged operationscreate project access tokens

Policy authors can write rules targeting input.resource.type == "urn:connector:identos:gitlab". Per-project rules use the resource_id (gitlab://projects/{project_id}).

Setup fields

FieldRequiredDefaultPurpose
base_urlyeshttps://gitlab.comGitLab base URL (SaaS or self-hosted).
upstream_auth.typeyesidp_passthroughAuthentication mode — idp_passthrough (per-user, default) or static (shared token).
token_envconditional(static mode only) Name of the env var holding the access token (not the token itself). Hidden when idp_passthrough is selected.

MCP tools

When the managed gateway's MCP aggregator is enabled, this connector exposes the following tools to agents:

ToolScopeDescription
list_usersgitlab:readSearch GitLab users.
list_projectsgitlab:readList projects (optionally filter by search, membership).
get_projectgitlab:readGet a project by ID or URL-encoded path.
list_merge_requestsgitlab:readList MRs on a project.
create_merge_requestgitlab:writeOpen a new MR.
list_merge_request_notesgitlab:readList comments on an MR.
create_merge_request_notegitlab:writeAdd a comment to an MR.
edit_merge_requestgitlab:writeUpdate MR title/description/state.
create_project_access_tokengitlab:adminMint a project access token.
list_registry_repositoriesgitlab:readList container registry repositories on a project.
list_registry_tagsgitlab:readList tags in a registry repository.

Each tool's input schema is embedded in the manifest and forwarded verbatim to MCP clients.

Example policy

Block MR creation on a sensitive project unless the caller belongs to the release-managers group:

package pbac.operator.connectors["identos.gitlab"]

deny contains msg if {
input.resource.type == "urn:connector:identos:gitlab"
input.action.method == "POST"
endswith(input.action.path, "/merge_requests")
input.resource.id == "gitlab://projects/release-critical"
not "release-managers" in input.subject.groups
msg := "only release-managers can open MRs against release-critical"
}

This rule lives in the operator namespace (not this connector's) because "which group is allowed" is tenant-specific.

Reference

Manifest reference

  • ID: identos.gitlab
  • Version: 1.0.0
  • Resource type: urn:connector:identos:gitlab
  • Capabilities: mcp

Supported auth modes

TypeDetails
staticscheme header; header PRIVATE-TOKEN; setup fields: token_env

Setup fields

IDLabelDefaultSecret?Notes
base_urlGitLab base URLhttps://gitlab.comnoplaceholder: https://git.example.com
token_envAccess tokenyesPick a secret holding a GitLab personal, project, or group access token.

Scopes

Scope
gitlab:read
gitlab:write
gitlab:admin

Routes

MethodPatternScopeResource template
GET/api/v4/usersgitlab:read
GET/api/v4/projectsgitlab:read
GET/api/v4/projects/{project_id}gitlab:readgitlab://projects/{{project_id}}
GET/api/v4/projects/{project_id}/merge_requestsgitlab:readgitlab://projects/{{project_id}}
POST/api/v4/projects/{project_id}/merge_requestsgitlab:writegitlab://projects/{{project_id}}
GET/api/v4/projects/{project_id}/merge_requests/{merge_request_iid}/notesgitlab:readgitlab://projects/{{project_id}}/mr/{{merge_request_iid}}
POST/api/v4/projects/{project_id}/merge_requests/{merge_request_iid}/notesgitlab:writegitlab://projects/{{project_id}}/mr/{{merge_request_iid}}
PUT/api/v4/projects/{project_id}/merge_requests/{merge_request_iid}gitlab:writegitlab://projects/{{project_id}}/mr/{{merge_request_iid}}
POST/api/v4/projects/{project_id}/access_tokensgitlab:admingitlab://projects/{{project_id}}
GET/api/v4/projects/{project_id}/registry/repositoriesgitlab:readgitlab://projects/{{project_id}}
GET/api/v4/projects/{project_id}/registry/repositories/{repository_id}/tagsgitlab:readgitlab://projects/{{project_id}}/registry/{{repository_id}}
GET/api/v4/projects/{project_id}/releasesgitlab:readgitlab://projects/{{project_id}}
GET/api/v4/projects/{project_id}/releases/{tag_name}gitlab:readgitlab://projects/{{project_id}}/releases/{{tag_name}}

MCP tools

NameScopeDescription
list_usersgitlab:readList GitLab users (optional search).
list_projectsgitlab:readList GitLab projects.
get_projectgitlab:readGet a GitLab project by ID or URL-encoded path.
list_merge_requestsgitlab:readList merge requests for a project.
create_merge_requestgitlab:writeCreate a merge request.
list_merge_request_notesgitlab:readList comments on a merge request.
create_merge_request_notegitlab:writeAdd a comment on a merge request.
edit_merge_requestgitlab:writeUpdate merge request title, description, or state.
create_project_access_tokengitlab:adminCreate a project access token.
list_registry_repositoriesgitlab:readList container registry repositories for a project.
list_registry_tagsgitlab:readList tags for a container registry repository.
list_releasesgitlab:readList releases for a project (newest first).
get_releasegitlab:readGet a single release by tag name.