Cells: Routing Service
This document describes design goals and architecture of Routing Service used by Cells. To better understand where the Routing Service fits into architecture take a look at Infrastructure Architecture.
Goals
The routing layer is meant to offer a consistent user experience where all Cells are presented under a single domain (for example, gitlab.com), instead of having to go to separate domains.
The user will be able to use https://gitlab.com to access Cell-enabled GitLab.
Depending on the URL access, it will be transparently proxied to the correct Cell that can serve this particular information.
For example:
- All requests going to https://gitlab.com/users/sign_inare randomly distributed to all Cells.
- All requests going to https://gitlab.com/gitlab-org/gitlab/-/tree/masterare always directed to Cell 5, for example.
- All requests going to https://gitlab.com/my-username/my-projectare always directed to Cell 1.
- 
Technology. We decide what technology the routing service is written in. The choice is dependent on the best performing language, and the expected way and place of deployment of the routing layer. If it is required to make the service multi-cloud it might be required to deploy it to the CDN provider. Then the service needs to be written using a technology compatible with the CDN provider. 
- 
Cell discovery. The routing service needs to be able to discover and monitor the health of all Cells. 
- 
User can use single domain to interact with many Cells. The routing service will intelligently route all requests to Cells based on the resource being accessed versus the Cell containing the data. 
- 
Router endpoints classification. The stateless routing service will fetch and cache information about endpoints from one of the Cells. We need to implement a protocol that will allow us to accurately describe the incoming request (its fingerprint), so it can be classified by one of the Cells, and the results of that can be cached. We also need to implement a mechanism for negative cache and cache eviction. 
- 
GraphQL and other ambiguous endpoints. Most endpoints have a unique sharding key: the Organization, which directly or indirectly (via a Group or Project) can be used to classify endpoints. Some endpoints are ambiguous in their usage (they don't encode the sharding key), or the sharding key is stored deep in the payload. In these cases, we need to decide how to handle endpoints like /api/graphql.
- 
Small. The Routing Service is configuration-driven and rules-driven, and does not implement any business logic. The maximum size of the project source code in initial phase is 1_000 lines without tests. The reason for the hard limit is to make the Routing Service to not have any special logic, and could be rewritten into any technology in a matter of a few days. 
Requirements
| Requirement | Description | Priority | 
|---|---|---|
| Discovery | needs to be able to discover and monitor the health of all Cells. | high | 
| Security | only authorized cells can be routed to | high | 
| Single domain | for example GitLab.com | high | 
| Caching | can cache routing information for performance | high | 
| Low latency | 50 ms of increased latency | high | 
| Path-based | can make routing decision based on path | high | 
| Complexity | the routing service should be configuration-driven and small | high | 
| Rolling | the routing service works with Cells running mixed versions | high | 
| Feature Flags | features can be turned on, off, and % rollout | high | 
| Progressive Rollout | we can slowly rollout a change | medium | 
| Stateless | does not need database, Cells provide all routing information | medium | 
| Secrets-based | can make routing decision based on secret (for example JWT) | medium | 
| Observability | can use existing observability tooling | low | 
| Self-managed | can be eventually used by self-managed | low | 
| Regional | can route requests to different regions | low | 
Low Latency
The target latency for routing service should be less than 50 ms.
Looking at the urgency: high request we don't have a lot of headroom on the p50.
Adding an extra 50 ms allows us to still be in or SLO on the p95 level.
There is 3 primary entry points for the application; web, api, and git.
Each service is assigned a Service Level Indicator (SLI) based on latency using the apdex standard.
The corresponding Service Level Objectives (SLOs) for these SLIs require low latencies for large amount of requests.
It's crucial to ensure that the addition of the routing layer in front of these services does not impact the SLIs.
The routing layer is a proxy for these services, and we lack a comprehensive SLI monitoring system for the entire request flow (including components like the Edge network and Load Balancers) we use the SLIs for web, git, and api as a target.
The main SLI we use is the rails requests.
It has multiple satisfied targets (apdex) depending on the request urgency:
| Urgency | Duration in ms | 
|---|---|
| :high | 250 ms | 
| :medium | 500 ms | 
| :default | 1000 ms | 
| :low | 5000 ms | 
Analysis
The way we calculate the headroom we have is by using the following:
\mathrm{Headroom}\ {ms} = \mathrm{Satisfied}\ {ms} - \mathrm{Duration}\ {ms}web:
| Target Duration | Percentile | Headroom | 
|---|---|---|
| 5000 ms | p99 | 4000 ms | 
| 5000 ms | p95 | 4500 ms | 
| 5000 ms | p90 | 4600 ms | 
| 5000 ms | p50 | 4900 ms | 
| 1000 ms | p99 | 500 ms | 
| 1000 ms | p95 | 740 ms | 
| 1000 ms | p90 | 840 ms | 
| 1000 ms | p50 | 900 ms | 
| 500 ms | p99 | 0 ms | 
| 500 ms | p95 | 60 ms | 
| 500 ms | p90 | 100 ms | 
| 500 ms | p50 | 400 ms | 
| 250 ms | p99 | 140 ms | 
| 250 ms | p95 | 170 ms | 
| 250 ms | p90 | 180 ms | 
| 250 ms | p50 | 200 ms | 
Analysis was done in https://gitlab.com/gitlab-org/gitlab/-/issues/432934#note_1667993089
api:
| Target Duration | Percentile | Headroom | 
|---|---|---|
| 5000 ms | p99 | 3500 ms | 
| 5000 ms | p95 | 4300 ms | 
| 5000 ms | p90 | 4600 ms | 
| 5000 ms | p50 | 4900 ms | 
| 1000 ms | p99 | 440 ms | 
| 1000 ms | p95 | 750 ms | 
| 1000 ms | p90 | 830 ms | 
| 1000 ms | p50 | 950 ms | 
| 500 ms | p99 | 450 ms | 
| 500 ms | p95 | 480 ms | 
| 500 ms | p90 | 490 ms | 
| 500 ms | p50 | 490 ms | 
| 250 ms | p99 | 90 ms | 
| 250 ms | p95 | 170 ms | 
| 250 ms | p90 | 210 ms | 
| 250 ms | p50 | 230 ms | 
Analysis was done in https://gitlab.com/gitlab-org/gitlab/-/issues/432934#note_1669995479
git:
| Target Duration | Percentile | Headroom | 
|---|---|---|
| 5000 ms | p99 | 3760 ms | 
| 5000 ms | p95 | 4280 ms | 
| 5000 ms | p90 | 4430 ms | 
| 5000 ms | p50 | 4900 ms | 
| 1000 ms | p99 | 500 ms | 
| 1000 ms | p95 | 750 ms | 
| 1000 ms | p90 | 800 ms | 
| 1000 ms | p50 | 900 ms | 
| 500 ms | p99 | 280 ms | 
| 500 ms | p95 | 370 ms | 
| 500 ms | p90 | 400 ms | 
| 500 ms | p50 | 430 ms | 
| 250 ms | p99 | 200 ms | 
| 250 ms | p95 | 230 ms | 
| 250 ms | p90 | 240 ms | 
| 250 ms | p50 | 240 ms | 
Analysis was done in https://gitlab.com/gitlab-org/gitlab/-/issues/432934#note_1671385680
Non-Goals
Not yet defined.
Proposal
The Routing Service implements the following design guidelines:
- Simple:
- Routing service does not buffer requests.
- Routing service can only proxy to a single Cell based on request headers.
 
- Stateless:
- Routing service does not have permanent storage.
- Routing service uses multi-level cache: in-memory, external shared cache.
 
- Zero-trust:
- Routing service signs each request that is being proxied.
- The trust is established by using JWT token, or mutual authentication scheme.
- Cells can be available over public internet, as long as they follow the zero-trust model.
 
- Configuration-based:
- Routing service is configured with a static list of Cells.
- Routing service configuration is applied as part of service deployment.
 
- Rule-based:
- Routing service is deployed with a routing rules gathered from all Cells.
- Routing service does support rules lists generated by different versions of GitLab.
- rules allows to match by any criteria: header, content of the header, or route path.
 
- Agnostic:
- Routing service is not aware of high-level concepts like organizations.
- The classification is done per-specification provided in a rules, to find the sharding key.
- The sharding key result is cached.
- The single sharding key cached is used to handle many similar requests.
 
The following diagram shows how a user request routes through DNS to the Routing Service deployed as Cloudflare Worker and the router chooses a cell to send the request to.
graph TD;
    user((User));
    router[Routing Service];
    cell_us0{Cell US0};
    cell_us1{Cell US1};
    cell_eu0{Cell EU0};
    cell_eu1{Cell EU1};
    user-->router;
    router-->cell_eu0;
    router-->cell_eu1;
    router-->cell_us0;
    router-->cell_us1;
    subgraph Europe
        cell_eu0;
        cell_eu1;
    end
    subgraph United States
        cell_us0;
        cell_us1;
    endRouting rules
Each Cell will publish a precompiled list of routing rules that will be consumed by the Routing Service:
- The routing rules describe how to decode the request, find the sharding key, and make the routing decision.
- The routing rules are compiled during the deployment of the Routing Service.
- The deployment process fetches latest version of the routing rules from each Cell that is part of Routing Service configuration.
- The compilation process merges the routing rules from all Cells.
- The conflicting rules prevent routing service from being compiled / started.
- Each routing rule entry has a unique identifier to ease the merge.
- The Routing Service would be re-deployed only if the list of rules was changed, which shouldn't happen frequently, because we expect the majority of newly added endpoints to already adhere to the prior route rules.
 
- The configuration describes from which Cells the routing rules need to be fetched during deploy.
- The published routing rules might make routing decision based on the secret. For example, if the session cookie
or authentication token has prefix c100-all requests are to be forwarded to the given Cell.
- The Cell does publish routing rules at /api/v4/internal/cells/route_rules.json.
- The rules published by Cell only include endpoints that the particular Cell can process.
- The Cell might request to perform dynamic classification based on sharding key, by configuring
routing rules to call /api/v4/internal/cells/classify.
- The routing rules should use prefixas a way to speed up classification. During the compilation phase the routing service transforms all found prefixes into a decision tree to speed up any subsequent regex matches.
- Some of the prefixes need to be Cell independent, example Personal Access Tokens prefix need to be organization bound and not Cell bound. We want the ability to move an organization from 1 cell to another without changing the Personal Access Token or any other token.
- The routing rules is ideally compiled into source code to avoid expensive parsing and evaluation of the rules dynamically as part of deployment.
The routing rules JSON structure describes all matchers:
{
    "rules": [
        {
            "id": "<unique-identifier>",
            "cookies": {
                "<cookie_name>": {
                    "prefix": "<match-given-prefix>",
                    "match_regex": "<regex_match>"
                },
                "<cookie_name2>": {
                    "prefix": "<match-given-prefix>",
                    "match_regex": "<regex_match>"
                }
            },
            "headers": {
                "<header_name>": {
                    "prefix": "<match-given-prefix>",
                    "match_regex": "<regex_match>"
                },
                "<header_name2>": {
                    "prefix": "<match-given-prefix>",
                    "match_regex": "<regex_match>"
                },
            },
            "path": {
                "prefix": "<match-given-prefix>",
                "match_regex": "<regex_match>"
            },
            "method": ["<list_of_accepted_methods>"],
            // If many rules are matched, define which one wins
            "priority": 1000,
            // Accept request and proxy to the Cell in question
            "action": "proxy",
            // Classify request based on regex matching groups
            "action": "classify",
            "classify": {
                "keys": ["list_of_regex_match_capture_groups"]
            }
        }
    ]
}Example of the routing rules published by the Cell 100 that makes routing decision based session cookie, and secret. The high priority is assigned since the routing rules is secret-based, and should take precedence before all other matchers:
{
    "rules": [
        {
            "id": "t4mkd5ndsk58si6uwwz7rdavil9m2hpq",
            "cookies": {
                "_gitlab_session": {
                    "prefix": "c100-" // accept `_gitlab_session` that are prefixed with `c100-`
                }
            },
            "action": "proxy",
            "priority": 1000
        },
        {
            "id": "jcshae4d4dtykt8byd6zw1ecccl5dkts",
            "headers": {
                "GITLAB_TOKEN": {
                    "prefix": "C100_" // accept `GITLAB_TOKEN` that are prefixed with `C100_`
                }
            },
            "action": "proxy",
            "priority": 1000
        }
    ]
}Example of the routing rules published by all Cells that makes routing decision based on the path:
{
    "rules": [
        {
            "id": "c9scvaiwj51a75kzoh917uwtnw8z4ebl",
            "path": {
                "prefix": "/api/v4/projects/", // speed-up rule matching
                "match_regex": "^/api/v4/projects/(?<project_id_or_path_encoded>[^/]+)(/.*)?$"
            },
            "action": "classify",
            "classify": {
                "keys": ["project_id_or_path_encoded"]
            }
        }
    ]
}Classification
Each Cell does implement classification endpoint:
- The classification endpoint is at /api/v4/internal/cells/classify(or gRPC endpoint).
- The classification endpoint accepts a list of the sharding keys. Sharding keys are decoded from request, based on the routing rules provided by the Cell.
- The endpoint returns other equivalent sharding keys to pollute cache for similar requests. This is to ensure that all similar requests can be handled quickly without having to classify each time.
- Routing Service tracks the health of Cells, and issues a classifyrequest to Cells based on weights, health of the Cell, or other defined criteria. Weights would indicate which Cell is preferred to perform the classification of sharding keys.
- Routing Service retries the classifycall for a reasonable amount of time. The repetitive failure of Cell toclassifyis indicative of Cell being unhealthy.
- The classifyresult is cached regardless of returnedaction(proxy or reject). The rejected classification is cached to prevent excessive amount of requests for sharding keys that are not found.
- The cached response is for time defined by expiryandrefresh.- The expirydefines when the item is removed from cache unless used.
- The refreshdefines when the item needs to be reclassified if used.
- The refresh is done asynchronously as the request should be served without a delay if they were classified. The refresh is done to ensure that cache is always hot and up-to date.
 
- The 
For the above example:
- The router sees request to /api/v4/projects/1000/issues.
- It selects the above rulefor this request, which requestsclassifyforproject_id_or_path_encoded.
- It decodes project_id_or_path_encodedto be1000.
- Checks the cache if there's project_id_or_path_encoded=1000associated to any Cell.
- Sends the request to /api/v4/internal/cells/classifyif no Cells was found in cache.
- Rails responds with the Cell holding the given project, and also all other equivalent sharding keys for the resource that should be put in the cache.
- Routing Service caches for the duration specified in configuration, or response.
# POST /api/v4/internal/cells/classify
## Request:
{
    "metadata": {
        "rule_id": "c9scvaiwj51a75kzoh917uwtnw8z4ebl",
        "headers": {
            "all_request_headers": "value"
        },
        "method": "GET",
        "path": "/api/v4/projects/100/issues"
    },
    "keys": {
        "project_id_or_path_encoded": 100
    }
}
## Response:
{
    "action": "proxy",
    "proxy": {
        "name": "cell_1",
        "url": "https://cell1.gitlab.com"
    },
    "ttl": "10 minutes",
    "matched_keys": [ // list of all equivalent keys that should be put in the cache
        { "project_id_or_path_encoded": 100 },
        { "project_id_or_path_encoded": "gitlab-org%2Fgitlab" },
        { "project_full_path": "gitlab-org/gitlab" },
        { "namespace_full_path": "gitlab-org" },
        { "namespace_id": 10 },
        { "organization_full_path": "gitlab-inc" },
        { "organization_id": 50 },
    ]
}The following code represents a negative response when a sharding key was not found:
# POST /api/v4/internal/cells/classify
## Request:
{
    "metadata": {
        "rule_id": "c9scvaiwj51a75kzoh917uwtnw8z4ebl",
        "headers": {
            "all_request_headers": "value"
        },
        "method": "GET",
        "path": "/api/v4/projects/100/issues"
    },
    "keys": {
        "project_id_or_path_encoded": 100
    }
}
## Response:
{
    "action": "reject",
    "reject": {
        "http_status": 404
    },
    "cache": {
        "refresh": "10 minutes",
        "expiry": "10 minutes"
    },
    "matched_keys": [ // list of all equivalent keys that should be put in the cache
        { "project_id_or_path_encoded": 100 },
    ]
}Configuration
The Routing Service will use the configuration similar to this:
[[cells]]
name=cell_1
url=https://cell1.gitlab.com
key=ABC123
classify_weight=100
[[cells]]
name=cell_2
url=https://cell2.gitlab.com
key=CDE123
classify_weight=1
[cache.memory.classify]
refresh_time=10 minutes
expiry_time=1 hour
[cache.external.classify]
refresh_time=30 minutes
expiry_time=6 hourWe assume that this is acceptable to provide a static list of Cells, because:
- Static: Cells provisioned are unlikely to be dynamically provisioned and decommissioned.
- Good enough: We can manage such list even up to 100 Cells.
- Simple: We don't have to implement robust service discovery in the service, and we have guarantee that this list is always exhaustive.
The configuration describes all Cells, URLs, zero-trust keys, and weights,
and how long requests should be cached. The classify_weight defines how often
the Cell should receive classification requests versus other Cells.
Request flows
- There are two Cells.
- 
gitlab-orgis a top-level namespace and lives inCell US0in theGitLab.com Publicorganization.
- 
my-companyis a top-level namespace and lives inCell EU0in themy-organizationorganization.
Router configured to perform static routing
- The Cell US0 supports all other public-facing projects.
- The Cells is configured to generate all secrets and session cookies with a prefix like eu0_for Cell EU0.- The Personal Access Token is scoped to Organization, and because the Organization is part only of a single Cell, the PATs generated are prefixed with Cell identifier.
- The Session Cookie encodes Organization in-use, and because the Organization is part only of a single Cell, the session cookie generated is prefixed with Cell identifier.
 
- The Cell EU0 allows only private organizations, groups, and projects.
- The Cell US0 is a target Cell for all requests unless explicitly prefixed.
Cell US0:
{
    "rules": [
        {
            "id": "tjh147se67wadjzum7onwqiad2b75uft",
            "path": {
                "prefix": "/"
            },
            "action": "proxy",
            "priority": 1
        }
    ]
}Cell EU0:
{
    "rules": [
        {
            "id": "t4mkd5ndsk58si6uwwz7rdavil9m2hpq",
            "cookies": {
                "_gitlab_session": {
                    "prefix": "eu0_"
                }
            },
            "path": {
                "prefix": "/"
            },
            "action": "proxy",
            "priority": 1000
        },
        {
            "id": "jcshae4d4dtykt8byd6zw1ecccl5dkts",
            "headers": {
                "GITLAB_TOKEN": {
                    "prefix": "eu0_"
                }
            },
            "path": {
                "prefix": "/"
            },
            "action": "proxy",
            "priority": 1000
        }
    ]
}
Goes to /my-company/my-project while logged in into Cell EU0
- Because user switched the Organization to my-company, its session cookie is prefixed witheu0_.
- User sends request /my-company/my-project, and because the cookie is prefixed witheu0_it is directed to Cell EU0.
- 
Cell EU0returns the correct response.
sequenceDiagram
    participant user as User
    participant router as Router
    participant cell_eu0 as Cell EU0
    participant cell_eu1 as Cell EU1
    user->>router: GET /my-company/my-project<br/>_gitlab_session=eu0_uwwz7rdavil9
    router->>cell_eu0: GET /my-company/my-project
    cell_eu0->>user: <h1>My Project...
Goes to /my-company/my-project while not logged in
- User visits /my-company/my-project, and because it does not have session cookie, the request is forwarded toCell US0.
- User signs in.
- GitLab sees that user default organization is my-company, so it assigns session cookie witheu0_to indicate that user is meant to interact withmy-company.
- User sends request to /my-company/my-projectagain, now with the session cookie that proxies toCell EU0.
- 
Cell EU0returns the correct response.
sequenceDiagram
    participant user as User
    participant router as Router
    participant cell_us0 as Cell US0
    participant cell_eu0 as Cell EU0
    user->>router: GET /my-company/my-project
    router->>cell_us0: GET /my-company/my-project
    cell_us0->>user: HTTP 302 /users/sign_in?redirect=/my-company/my-project
    user->>router: GET /users/sign_in?redirect=/my-company/my-project
    router->>cell_us0: GET /users/sign_in?redirect=/my-company/my-project
    cell_us0-->>user: <h1>Sign in...
    user->>router: POST /users/sign_in?redirect=/my-company/my-project
    router->>cell_us0: POST /users/sign_in?redirect=/my-company/my-project
    cell_us0->>user: HTTP 302 /my-company/my-project<br/>_gitlab_session=eu0_uwwz7rdavil9
    user->>router: GET /my-company/my-project<br/>_gitlab_session=eu0_uwwz7rdavil9
    router->>cell_eu0: GET /my-company/my-project<br/>_gitlab_session=eu0_uwwz7rdavil9
    cell_eu0->>user: <h1>My Project...
Goes to /gitlab-org/gitlab after last step
User visits /my-company/my-project, and because it does not have a session cookie, the request is forwarded to Cell US0.
sequenceDiagram
    participant user as User
    participant router as Router
    participant cell_eu0 as Cell EU0
    participant cell_us0 as Cell US0
    user->>router: GET /gitlab-org/gitlab<br/>_gitlab_session=eu0_uwwz7rdavil9
    router->>cell_eu0: GET /gitlab-org/gitlab
    cell_eu0->>user: HTTP 404Router configured to perform dynamic routing based on classification
The Cells publish route rules that allows to classify the requests.
Cell US0 and EU0:
{
    "rules": [
        {
            "id": "tjh147se67wadjzum7onwqiad2b75uft",
            "path": {
                "prefix": "/",
                "regex": "^/(?top_level_group)[^/]+(/.*)?$",
            },
            "action": "classify",
            "classify": {
                "keys": ["top_level_group"]
            }
        },
        {
            "id": "jcshae4d4dtykt8byd6zw1ecccl5dkts",
            "path": {
                "prefix": "/"
            },
            "action": "proxy"
        }
    ]
}
Goes to /my-company/my-project while logged in into Cell EU0
- The /my-company/my-project/is visited.
- Router decodes sharding key top_level_group=my-company.
- Router checks if this sharding key is cached.
- Because it is not, the classification request is sent to a random Cell to /classify.
- The response of classify is cached.
- The request is then proxied to Cell returned by classification.
sequenceDiagram
    participant user as User
    participant router as Router
    participant cache as Cache
    participant cell_us0 as Cell US0
    participant cell_eu0 as Cell EU0
    user->>router: GET /my-company/my-project
    router->>cache: CACHE_GET: top_level_group=my-company
    cache->>router: CACHE_NOT_FOUND
    router->>cell_us0: POST /api/v4/internal/cells/classify<br/>top_level_group=my-company
    cell_us0->>router: CLASSIFY: top_level_group=my-company, cell=cell_eu0
    router->>cache: CACHE_SET: top_level_group=my-company, cell=cell_eu0
    router->>cell_eu0: GET /my-company/my-project
    cell_eu0->>user: <h1>My Project...
Goes to /my-company/my-project while not logged in
- The /my-company/my-project/is visited.
- Router decodes sharding key top_level_group=my-company.
- Router checks if this sharding key is cached.
- Because it is not, the classification request is sent to a random Cell to /classify.
- The response of classifyis cached.
- The request is then proxied to Cell returned by classification.
- Because project is private, user is redirected to sign in.
- The sign-in since is defined to be handled by all Cells, so it is proxied to a random Cell.
- User visits the /my-company/my-project/again after logging in.
- The top_level_group=my-companyis proxied to the correct Cell.
sequenceDiagram
    participant user as User
    participant router as Router
    participant cache as Cache
    participant cell_us0 as Cell US0
    participant cell_eu0 as Cell EU0
    user->>router: GET /my-company/my-project
    router->>cache: CACHE_GET: top_level_group=my-company
    cache->>router: CACHE_NOT_FOUND
    router->>cell_us0: POST /api/v4/internal/cells/classify<br/>top_level_group=my-company
    cell_us0->>router: CLASSIFY: top_level_group=my-company, cell=cell_eu0
    router->>cache: CACHE_SET: top_level_group=my-company, cell=cell_eu0
    router->>cell_eu0: GET /my-company/my-project
    cell_eu0->>user: HTTP 302 /users/sign_in?redirect=/my-company/my-project
    user->>router: GET /users/sign_in?redirect=/my-company/my-project
    router->>cell_us0: GET /users/sign_in?redirect=/my-company/my-project
    cell_us0-->>user: <h1>Sign in...
    user->>router: POST /users/sign_in?redirect=/my-company/my-project
    router->>cell_eu0: POST /users/sign_in?redirect=/my-company/my-project
    cell_eu0->>user: HTTP 302 /my-company/my-project
    user->>router: GET /my-company/my-project
    router->>cache: CACHE_GET: top_level_group=my-company
    cache->>router: CACHE_FOUND: cell=cell_eu0
    router->>cell_eu0: GET /my-company/my-project
    cell_eu0->>user: <h1>My Project...
Goes to /gitlab-org/gitlab after last step
- Because the /gitlab-orgis not found in cache, it will be classified and then directed to correct Cell.
sequenceDiagram
    participant user as User
    participant router as Router
    participant cache as Cache
    participant cell_us0 as Cell US0
    participant cell_eu0 as Cell EU0
    user->>router: GET /gitlab-org/gitlab
    router->>cache: CACHE_GET: top_level_group=gitlab-org
    cache->>router: CACHE_NOT_FOUND
    router->>cell_us0: POST /api/v4/internal/cells/classify<br/>top_level_group=gitlab-org
    cell_us0->>router: CLASSIFY: top_level_group=gitlab-org, cell=cell_us0
    router->>cache: CACHE_SET: top_level_group=gitlab-org, cell=cell_us0
    router->>cell_us0: GET /gitlab-org/gitlab
    cell_us0->>user: <h1>My Project...Performance and reliability considerations
- It is expected that each Cell can classify all sharding keys.
- Alternatively the classification could be done by Cluster-wide Data Provider if it would own all data required to classify.
- The published routing rules allow to define static criteria, allowing to make routing decision only on a secret. As a result, the Routing Service doesn't add any latency for request processing, and superior resiliency.
- It is expected that there will be penalty when learning new sharding key. However, it is expected that multi-layer cache should provide a very high cache-hit-ratio, due to low cardinality of sharding key. The sharding key would effectively be mapped into resource (organization, group, or project), and there's a finite amount of those.
Alternatives
Buffering requests
The Stateless Router using Requests Buffering
describes an approach where Cell answers with X-Gitlab-Cell-Redirect to redirect request to another Cell:
- This is based on a need to buffer the whole request (headers + body) which is very memory intensive.
- This proposal does not provide an easy way to handle mixed deployment of Cells, where Cells might be running different versions.
- This proposal likely requires caching significantly more information, since it is based on requests, rather than on decoded sharding keys.
Learn request
The Stateless Router using Routes Learning
describes an approach similar to the one in this document. Except the route rules and classification
is done in a single go in a form of pre-flight check /api/v4/internal/cells/learn:
- This makes the whole routes learning dynamic, and dependent on availability of the Cells.
- This proposal does not provide an easy way to handle mixed deployment of Cells, where Cells might be running different versions.
- This proposal likely requires caching significantly more information, since it is based on requests, rather than on decoded sharding keys.
FAQ
- How and when will Routing Service compile set of rules?
To be defined.