Commit 79b13a8d authored by GitLab Bot's avatar GitLab Bot

Automatic merge of gitlab-org/gitlab-ce master

parents 4447c081 497ba83f
......@@ -65,6 +65,7 @@ description: 'Learn how to contribute to GitLab.'
- [Repository mirroring](repository_mirroring.md)
- [Git LFS](lfs.md)
- [Developing against interacting components or features](interacting_components.md)
- [File uploads](uploads.md)
## Performance guides
......
......@@ -2,6 +2,8 @@
We use the [CarrierWave] gem to handle file upload, store and retrieval.
File uploads should be accelerated by workhorse, for details please refer to [uploads development documentation](uploads.md).
There are many places where file uploading is used, according to contexts:
- System
......
# Uploads development documentation
[GitLab Workhorse](https://gitlab.com/gitlab-org/gitlab-workhorse) has special rules for handling uploads.
To prevent occupying a ruby process on I/O operations, we process the upload in workhorse, where is cheaper.
This process can also directly upload to object storage.
## The problem description
The following graph explains machine boundaries in a scalable GitLab installation. Without any workhorse optimization in place, we can expect incoming requests to follow the numbers on the arrows.
```mermaid
graph TB
subgraph "load balancers"
LB(HA Proxy)
end
subgraph "Shared storage"
nfs(NFS)
end
subgraph "redis cluster"
r(persisted redis)
end
LB-- 1 -->workhorse
subgraph "web or API fleet"
workhorse-- 2 -->rails
end
rails-- "3 (write files)" -->nfs
rails-- "4 (schedule a job)" -->r
subgraph sidekiq
s(sidekiq)
end
s-- "5 (fetch a job)" -->r
s-- "6 (read files)" -->nfs
```
We have three challenges here: performance, availability, and scalability.
### Performance
Rails process are expensive in terms of both CPU and memory. Ruby [global interpreter lock](https://en.wikipedia.org/wiki/Global_interpreter_lock) adds to cost too because the ruby process will spend time on I/O operations on step 3 causing incoming requests to pile up.
In order to improve this, [workhorse disk acceleration](#workhorse-disk-acceleration) was implemented. With this, Rails no longer deals with writing uploaded files to disk.
```mermaid
graph TB
subgraph "load balancers"
LB(HA Proxy)
end
subgraph "Shared storage"
nfs(NFS)
end
subgraph "redis cluster"
r(persisted redis)
end
LB-- 1 -->workhorse
subgraph "web or API fleet"
workhorse-- "3 (without files)" -->rails
end
workhorse -- "2 (write files)" -->nfs
rails-- "4 (schedule a job)" -->r
subgraph sidekiq
s(sidekiq)
end
s-- "5 (fetch a job)" -->r
s-- "6 (read files)" -->nfs
```
### Availability
There's also an availability problem in this setup, NFS is a [single point of failure](https://en.wikipedia.org/wiki/Single_point_of_failure).
To address this problem an HA object storage can be used and it's supported by [workhorse object storage acceleration](#workhorse-object-storage-acceleration)
### Scalability
Scaling NFS is outside of our support scope, and NFS is not a part of cloud native installations.
All features that require sidekiq and do not use object storage acceleration won't work without NFS. In Kubernetes, machine boundaries translate to PODs, and in this case the uploaded file will be written into the POD private disk. Since sidekiq POD cannot reach into other pods, the operation will fail to read it.
## How to select the proper level of acceleration?
Selecting the proper acceleration is a tradeoff between speed of development and operational costs.
We can identify three major use-cases for an upload:
1. **storage:** if we are uploading for storing a file (i.e. artifacts, packages, discussion attachments). In this case [object storage acceleration](#workhorse-object-storage-acceleration) is the proper level as it's the less resource-intensive operation. Additional information can be found on [File Storage in GitLab](file_storage.md).
1. **in-controller/synchronous processing:** if we allow processing **small files** synchronously, using [disk acceleration](#workhorse-disk-acceleration) may speed up development.
1. **sidekiq/asynchronous processing:** Async processing must implement [object storage acceleration](#workhorse-object-storage-acceleration), the reason being that it's the only way to support Cloud Native deployments without a shared NFS.
For more details about currently broken feature see [epic &1802](https://gitlab.com/groups/gitlab-org/-/epics/1802).
### Handling repository uploads
Some features involves git repository uploads without using a regular git client.
Some examples are uploading a repository file from the web interface and [design management](../user/project/issues/design_management.md).
Those uploads requires the rails controller to act as a git client in lieu of the user.
Those operation falls into _in-controller/synchronous processing_ category, but we have no warranties on the file size.
In case of a LFS upload, the file pointer is committed synchronously, but file upload to object storage is performed asynchronously with sidekiq.
## Upload encodings
By upload encoding we mean how the file is included within the incoming request.
We have three kinds of file encoding in our uploads:
1. <i class="fa fa-check-circle"></i> **multipart**: `multipart/form-data` is the most common, a file is encoded as a part of a multipart encoded request.
1. <i class="fa fa-check-circle"></i> **body**: some APIs uploads files as the whole request body.
1. <i class="fa fa-times-circle"></i> **JSON**: some JSON API uploads files as base64 encoded strings. This requires [gitlab-workhorse#226](https://gitlab.com/gitlab-org/gitlab-workhorse/issues/226) to be implemented.
## Uploading technologies
By uploading technologies we mean how all the involved services interact with each other.
GitLab supports 3 kinds of uploading technologies, here follows a brief description with a sequence diagram for each one. Diagrams are not meant to be exhaustive.
### Regular rails upload
This is the default kind of upload, and it's most expensive in terms of resources.
In this case, workhorse is unaware of files being uploaded and acts as a regular proxy.
When a multipart request reaches the rails application, `Rack::Multipart` leaves behind tempfiles in `/tmp` and uses valuable Ruby process time to copy files around.
```mermaid
sequenceDiagram
participant c as Client
participant w as Workhorse
participant r as Rails
activate c
c ->>+w: POST /some/url/upload
w->>+r: POST /some/url/upload
r->>r: save the incoming file on /tmp
r->>r: read the file for processing
r-->>-c: request result
deactivate c
deactivate w
```
### Workhorse disk acceleration
This kind of upload avoids wasting resources caused by handling upload writes to `/tmp` in rails.
This optimization is not active by default on REST API requests.
When enabled, Workhorse looks for files in multipart MIME requests, uploading
any it finds to a temporary file on shared storage. The MIME data in the request
is replaced with the path to the corresponding file before it is forwarded to
Rails.
To prevent abuse of this feature, Workhorse signs the modified request with a
special header, stating which entries it modified. Rails will ignore any
unsigned path entries.
```mermaid
sequenceDiagram
participant c as Client
participant w as Workhorse
participant r as Rails
participant s as NFS
activate c
c ->>+w: POST /some/url/upload
w->>+s: save the incoming file on a temporary location
s-->>-w:
w->>+r: POST /some/url/upload
Note over w,r: file was replaced with its location<br>and other metadata
opt requires async processing
r->>+redis: schedule a job
redis-->>-r:
end
r-->>-c: request result
deactivate c
w->>-w: cleanup
opt requires async processing
activate sidekiq
sidekiq->>+redis: fetch a job
redis-->>-sidekiq: job
sidekiq->>+s: read file
s-->>-sidekiq: file
sidekiq->>sidekiq: process file
deactivate sidekiq
end
```
### Workhorse object storage acceleration
This is the more advanced acceleration technique we have in place.
Workhorse asks rails for temporary pre-signed object storage URLs and directly uploads to object storage.
In this setup an extra rails route needs to be implemented in order to handle authorization,
you can see an example of this in [`Projects::LfsStorageController`](https://gitlab.com/gitlab-org/gitlab-ce/blob/master/app/controllers/projects/lfs_storage_controller.rb)
and [its routes](https://gitlab.com/gitlab-org/gitlab-ce/blob/v12.2.0/config/routes/git_http.rb#L31-32).
**note:** this will fallback to _Workhorse disk acceleration_ when object storage is not enabled in the gitlab instance. The answer to the `/authorize` call will only contain a file system path.
```mermaid
sequenceDiagram
participant c as Client
participant w as Workhorse
participant r as Rails
participant os as Object Storage
activate c
c ->>+w: POST /some/url/upload
w ->>+r: POST /some/url/upload/authorize
Note over w,r: this request has an empty body
r-->>-w: presigned OS URL
w->>+os: PUT file
Note over w,os: file is stored on a temporary location. Rails select the destination
os-->>-w:
w->>+r: POST /some/url/upload
Note over w,r: file was replaced with its location<br>and other metadata
r->>+os: move object to final destination
os-->>-r:
opt requires async processing
r->>+redis: schedule a job
redis-->>-r:
end
r-->>-c: request result
deactivate c
w->>-w: cleanup
opt requires async processing
activate sidekiq
sidekiq->>+redis: fetch a job
redis-->>-sidekiq: job
sidekiq->>+os: get object
os-->>-sidekiq: file
sidekiq->>sidekiq: process file
deactivate sidekiq
end
```
## What does the `direct_upload` setting mean?
[Object storage setting](../administration/uploads.md#object-storage-settings) allows instance administators to enable `direct_upload`, this in an option that only affects the behavior of [workhorse object storage acceleration](#workhorse-object-storage-acceleration).
This option affect the response to the `/authorize` call. When not enabled, the API response will not contain presigned URLs and workhorse will write the file the shared disk, on the path is provided by rails, acting like object storage was disabled.
Once the request reachs rails, it will schedule an object storage upload as a sidekiq job.
......@@ -97,7 +97,7 @@ module Gitlab
attr_reader :load_times_by_model, :private_token
def debug(message, *)
message.gsub!(private_token, FILTERED_STRING) if private_token
message = message.gsub(private_token, FILTERED_STRING) if private_token
_, type, time = *message.match(/(\w+) Load \(([0-9.]+)ms\)/)
......
......@@ -17,8 +17,9 @@ module Gitlab
return unless content
all_names.each do |a_name|
content.gsub!(%r{/#{a_name} ?(.*)$}i, execute_block(action_block, context, '\1'))
content = content.gsub(%r{/#{a_name} ?(.*)$}i, execute_block(action_block, context, '\1'))
end
content
end
end
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::LegacyGithubImport::ReleaseFormatter do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::LegacyGithubImport::UserFormatter do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::LegacyGithubImport::WikiFormatter do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::LoopHelpers do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ManifestImport::Manifest do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ManifestImport::ProjectCreator do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::MarkupHelper do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::BackgroundTransaction do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Delta do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Instrumentation do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::MethodCall do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Methods do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Metric do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Prometheus, :prometheus do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::RackMiddleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::RequestsRackMiddleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Samplers::InfluxSampler do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Samplers::RubySampler do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Samplers::UnicornSampler do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::SidekiqMetricsExporter do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::SidekiqMiddleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Subscribers::ActionView do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Subscribers::ActiveRecord do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::Subscribers::RailsCache do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::System do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics::WebTransaction do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Metrics do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Middleware::BasicHealthCheck do
......
# frozen_string_literal: true
require 'spec_helper'
require 'tempfile'
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Middleware::RailsQueueDuration do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Middleware::ReadOnly do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Middleware::ReleaseEnv do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::MultiCollectionPaginator do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ObjectHierarchy do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Octokit::Middleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::OmniauthInitializer do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::OptimisticLocking do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::OtherMarkup do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::OtpKeyRotator do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PagesClient do
......
# coding: utf-8
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PathRegex do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PerformanceBar do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PhabricatorImport::Importer do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PhabricatorImport::UserFinder, :clean_gitlab_redis_cache do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PhabricatorImport::WorkerState, :clean_gitlab_redis_shared_state do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Plugin do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PollingInterval do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Popen::Runner do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Popen do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Profiler do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ProjectAuthorizations do
......
# coding: utf-8
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ProjectSearchResults do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ProjectTemplate do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ProjectTransfer do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Prometheus::AdditionalMetricsParser do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Prometheus::Queries::AdditionalMetricsDeploymentQuery do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Prometheus::Queries::AdditionalMetricsEnvironmentQuery do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Prometheus::Queries::DeploymentQuery do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Prometheus::Queries::MatchedMetricQuery do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::PrometheusClient do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QueryLimiting::ActiveSupportSubscriber do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QueryLimiting::Middleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QueryLimiting::Transaction do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QueryLimiting do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QuickActions::CommandDefinition do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QuickActions::Dsl do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QuickActions::Extractor do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QuickActions::SpendTimeAndDateSeparator do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::QuickActions::SubstitutionDefinition do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Redis::Cache do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Redis::Queues do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Redis::SharedState do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Redis::Wrapper do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ReferenceCounter do
......
# coding: utf-8
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Regex do
......
# frozen_string_literal: true
require 'spec_helper'
describe ::Gitlab::RepoPath do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::RepositoryCacheAdapter do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::RepositoryCache do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::RepositorySetCache, :clean_gitlab_redis_cache do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::RequestContext do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::RequestForgeryProtection, :allow_forgery_protection do
......
# frozen_string_literal: true
require 'fast_spec_helper'
describe Gitlab::RequestProfiler::Profile do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::RequestProfiler do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::RouteMap do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Routing do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sanitizers::Exif do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sanitizers::SVG do
......
# coding: utf-8
# frozen_string_literal: true
require 'spec_helper'
......@@ -108,7 +109,7 @@ describe Gitlab::Search::FoundBlob do
end
context 'with ISO-8859-1' do
let(:search_result) { "master:encoding/iso8859.txt\x001\x00\xC4\xFC\nmaster:encoding/iso8859.txt\x002\x00\nmaster:encoding/iso8859.txt\x003\x00foo\n".force_encoding(Encoding::ASCII_8BIT) }
let(:search_result) { (+"master:encoding/iso8859.txt\x001\x00\xC4\xFC\nmaster:encoding/iso8859.txt\x002\x00\nmaster:encoding/iso8859.txt\x003\x00foo\n").force_encoding(Encoding::ASCII_8BIT) }
it 'returns results as UTF-8' do
expect(subject.filename).to eq('encoding/iso8859.txt')
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Search::Query do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SearchResults do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sentry do
......
# frozen_string_literal: true
require 'fast_spec_helper'
describe Gitlab::Serializer::Ci::Variables do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Serializer::Pagination do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::ShardHealthCache, :clean_gitlab_redis_cache do
......
# frozen_string_literal: true
require 'spec_helper'
require 'stringio'
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::Collection do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::FileSample do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::LineProfiler do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::LineSample do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::Location do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::Middleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::Query do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Sherlock::Transaction do
......
# frozen_string_literal: true
require 'rails_helper'
describe Gitlab::SidekiqConfig do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqLogging::JSONFormatter do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqLogging::StructuredLogger do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqMiddleware::MemoryKiller do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqSignals do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqStatus::ClientMiddleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqStatus::ServerMiddleware do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqStatus do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqVersioning::Manager do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SidekiqVersioning, :sidekiq, :redis do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Command do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Deploy do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::IssueMove, service: true do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::IssueNew do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::IssueSearch do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::IssueShow do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Presenters::Access do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Presenters::Deploy do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Presenters::IssueMove do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Presenters::IssueNew do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Presenters::IssueSearch do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SlashCommands::Presenters::IssueShow do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SnippetSearchResults do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SQL::CTE do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SQL::Glob do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SQL::Pattern do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SQL::RecursiveCTE do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SQL::Union do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::SSHPublicKey, lib: true do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::StringPlaceholderReplacer do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::StringRangeMarker do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::StringRegexMarker do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::TcpChecker do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Template::Finders::GlobalTemplateFinder do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Template::Finders::RepoTemplateFinder do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Template::GitignoreTemplate do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Template::IssueTemplate do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Template::MergeRequestTemplate do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Themes, lib: true do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::TreeSummary do
......
# frozen_string_literal: true
require 'fast_spec_helper'
require 'support/shared_examples/malicious_regexp_shared_examples'
require 'support/helpers/stub_feature_flags'
......
# frozen_string_literal: true
require 'fast_spec_helper'
require 'support/shared_examples/malicious_regexp_shared_examples'
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::UploadsTransfer do
......
# coding: utf-8
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::UrlBlocker do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::UrlBuilder do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::UrlSanitizer do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::UserAccess do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Utils::DeepSize do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Utils::MergeHash do
describe '.crush' do
......
# frozen_string_literal: true
require 'fast_spec_helper'
describe Gitlab::Utils::Override do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Utils::SanitizeNodeLink do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Utils::StrongMemoize do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Utils do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Verify::JobArtifacts do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Verify::LfsObjects do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Verify::Uploads do
......
# frozen_string_literal: true
require 'spec_helper'
describe 'Gitlab::VersionInfo' do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::View::Presenter::Base do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::View::Presenter::Delegated do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::View::Presenter::Factory do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::View::Presenter::Simple do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::VisibilityLevel do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::WikiFileFinder do
......
# frozen_string_literal: true
require 'spec_helper'
describe Gitlab::Workhorse do
......
# frozen_string_literal: true
require 'fast_spec_helper'
require_dependency 'gitlab'
......
# frozen_string_literal: true
require 'spec_helper'
describe GoogleApi::Auth do
......
# frozen_string_literal: true
require 'spec_helper'
describe GoogleApi::CloudPlatform::Client do
......
# frozen_string_literal: true
describe JSONWebToken::RSAToken do
let(:rsa_key) do
OpenSSL::PKey::RSA.new <<-eos.strip_heredoc
......
# frozen_string_literal: true
describe JSONWebToken::Token do
let(:token) { described_class.new }
......
# frozen_string_literal: true
require 'spec_helper'
describe Mattermost::Client do
......
# frozen_string_literal: true
require 'spec_helper'
describe Mattermost::Command do
......
# frozen_string_literal: true
require 'spec_helper'
describe Mattermost::Session, type: :request do
......
# frozen_string_literal: true
require 'spec_helper'
describe Mattermost::Team do
......
# frozen_string_literal: true
require 'spec_helper'
describe MicrosoftTeams::Activity do
......
# frozen_string_literal: true
require 'spec_helper'
describe MicrosoftTeams::Notifier do
......
# frozen_string_literal: true
require 'spec_helper'
describe MilestoneArray do
......
# frozen_string_literal: true
require 'spec_helper'
describe ObjectStorage::DirectUpload do
......
# frozen_string_literal: true
require 'spec_helper'
describe OmniAuth::Strategies::Jwt do
......
# frozen_string_literal: true
require 'spec_helper'
describe RspecFlaky::Config, :aggregate_failures do
......
# frozen_string_literal: true
require 'spec_helper'
describe RspecFlaky::Example do
......
# frozen_string_literal: true
require 'spec_helper'
describe RspecFlaky::FlakyExample, :aggregate_failures do
......
# frozen_string_literal: true
require 'spec_helper'
describe RspecFlaky::FlakyExamplesCollection, :aggregate_failures do
......
# frozen_string_literal: true
require 'spec_helper'
describe RspecFlaky::Listener, :aggregate_failures do
......
# frozen_string_literal: true
require 'spec_helper'
describe RspecFlaky::Report, :aggregate_failures do
......
# frozen_string_literal: true
require 'spec_helper'
describe SafeZip::Entry do
......
# frozen_string_literal: true
require 'spec_helper'
describe SafeZip::ExtractParams do
......
# frozen_string_literal: true
require 'spec_helper'
describe SafeZip::Extract do
......
# frozen_string_literal: true
require 'fast_spec_helper'
describe Serializers::JSON do
......
# frozen_string_literal: true
require 'spec_helper'
describe SystemCheck::App::GitUserDefaultSSHConfigCheck do
......
# frozen_string_literal: true
require 'spec_helper'
describe SystemCheck::BaseCheck do
......
# frozen_string_literal: true
require 'spec_helper'
require 'rake_helper'
......
# frozen_string_literal: true
require 'spec_helper'
require 'rake_helper'
......
# frozen_string_literal: true
require 'spec_helper'
require 'rake_helper'
......
# frozen_string_literal: true
require 'spec_helper'
require 'rake_helper'
......
# frozen_string_literal: true
require 'spec_helper'
describe UploadedFile do
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment