Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
6f6c3c6f
Commit
6f6c3c6f
authored
Oct 03, 2018
by
Toon Claes
Committed by
Douglas Barbosa Alexandre
Oct 03, 2018
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Use Geo log to remove files when migrated to object storage
parent
a85c4cf4
Changes
40
Hide whitespace changes
Inline
Side-by-side
Showing
40 changed files
with
218 additions
and
158 deletions
+218
-158
app/uploaders/gitlab_uploader.rb
app/uploaders/gitlab_uploader.rb
+10
-0
app/uploaders/job_artifact_uploader.rb
app/uploaders/job_artifact_uploader.rb
+2
-0
app/uploaders/legacy_artifact_uploader.rb
app/uploaders/legacy_artifact_uploader.rb
+2
-0
app/uploaders/lfs_object_uploader.rb
app/uploaders/lfs_object_uploader.rb
+2
-0
app/uploaders/object_storage.rb
app/uploaders/object_storage.rb
+2
-0
ee/app/models/ee/ci/build.rb
ee/app/models/ee/ci/build.rb
+6
-0
ee/app/models/ee/ci/job_artifact.rb
ee/app/models/ee/ci/job_artifact.rb
+2
-4
ee/app/models/ee/lfs_object.rb
ee/app/models/ee/lfs_object.rb
+2
-4
ee/app/models/ee/note.rb
ee/app/models/ee/note.rb
+1
-1
ee/app/models/ee/upload.rb
ee/app/models/ee/upload.rb
+2
-4
ee/app/services/geo/file_download_service.rb
ee/app/services/geo/file_download_service.rb
+3
-1
ee/app/services/geo/file_registry_removal_service.rb
ee/app/services/geo/file_registry_removal_service.rb
+10
-14
ee/app/services/geo/file_service.rb
ee/app/services/geo/file_service.rb
+14
-4
ee/app/services/geo/job_artifact_deleted_event_store.rb
ee/app/services/geo/job_artifact_deleted_event_store.rb
+1
-14
ee/app/services/geo/lfs_object_deleted_event_store.rb
ee/app/services/geo/lfs_object_deleted_event_store.rb
+1
-14
ee/app/services/geo/upload_deleted_event_store.rb
ee/app/services/geo/upload_deleted_event_store.rb
+0
-7
ee/app/uploaders/ee/object_storage.rb
ee/app/uploaders/ee/object_storage.rb
+17
-0
ee/changelogs/unreleased/tc-geo-log-object-storage.yml
ee/changelogs/unreleased/tc-geo-log-object-storage.yml
+5
-0
ee/lib/gitlab/geo/log_cursor/events/job_artifact_deleted_event.rb
...itlab/geo/log_cursor/events/job_artifact_deleted_event.rb
+7
-29
ee/lib/gitlab/geo/log_cursor/events/lfs_object_deleted_event.rb
.../gitlab/geo/log_cursor/events/lfs_object_deleted_event.rb
+3
-10
ee/lib/gitlab/geo/log_cursor/events/upload_deleted_event.rb
ee/lib/gitlab/geo/log_cursor/events/upload_deleted_event.rb
+6
-9
ee/spec/lib/gitlab/geo/log_cursor/events/job_artifact_deleted_event_spec.rb
.../geo/log_cursor/events/job_artifact_deleted_event_spec.rb
+6
-3
ee/spec/lib/gitlab/geo/log_cursor/events/lfs_object_deleted_event_spec.rb
...ab/geo/log_cursor/events/lfs_object_deleted_event_spec.rb
+3
-5
ee/spec/lib/gitlab/geo/log_cursor/events/upload_deleted_event_spec.rb
...gitlab/geo/log_cursor/events/upload_deleted_event_spec.rb
+1
-1
ee/spec/models/ee/appearance_spec.rb
ee/spec/models/ee/appearance_spec.rb
+40
-0
ee/spec/models/ee/note_spec.rb
ee/spec/models/ee/note_spec.rb
+28
-0
ee/spec/services/geo/file_registry_removal_service_spec.rb
ee/spec/services/geo/file_registry_removal_service_spec.rb
+1
-1
ee/spec/services/geo/hashed_storage_attachments_event_store_spec.rb
...rvices/geo/hashed_storage_attachments_event_store_spec.rb
+1
-1
ee/spec/services/geo/hashed_storage_migrated_event_store_spec.rb
.../services/geo/hashed_storage_migrated_event_store_spec.rb
+1
-1
ee/spec/services/geo/job_artifact_deleted_event_store_spec.rb
...pec/services/geo/job_artifact_deleted_event_store_spec.rb
+5
-9
ee/spec/services/geo/lfs_object_deleted_event_store_spec.rb
ee/spec/services/geo/lfs_object_deleted_event_store_spec.rb
+4
-8
ee/spec/services/geo/repositories_changed_event_store_spec.rb
...pec/services/geo/repositories_changed_event_store_spec.rb
+1
-1
ee/spec/services/geo/repository_created_event_store_spec.rb
ee/spec/services/geo/repository_created_event_store_spec.rb
+1
-1
ee/spec/services/geo/repository_deleted_event_store_spec.rb
ee/spec/services/geo/repository_deleted_event_store_spec.rb
+1
-1
ee/spec/services/geo/repository_renamed_event_store_spec.rb
ee/spec/services/geo/repository_renamed_event_store_spec.rb
+1
-1
ee/spec/services/geo/repository_updated_event_store_spec.rb
ee/spec/services/geo/repository_updated_event_store_spec.rb
+1
-1
ee/spec/services/geo/reset_checksum_event_store_spec.rb
ee/spec/services/geo/reset_checksum_event_store_spec.rb
+1
-1
ee/spec/services/geo/upload_deleted_event_store_spec.rb
ee/spec/services/geo/upload_deleted_event_store_spec.rb
+4
-8
ee/spec/support/shared_examples/geo_event_store_shared_examples.rb
...upport/shared_examples/geo_event_store_shared_examples.rb
+12
-0
spec/support/shared_examples/uploaders/gitlab_uploader_shared_examples.rb
...red_examples/uploaders/gitlab_uploader_shared_examples.rb
+8
-0
No files found.
app/uploaders/gitlab_uploader.rb
View file @
6f6c3c6f
...
@@ -63,6 +63,12 @@ class GitlabUploader < CarrierWave::Uploader::Base
...
@@ -63,6 +63,12 @@ class GitlabUploader < CarrierWave::Uploader::Base
super
||
file
&
.
filename
super
||
file
&
.
filename
end
end
def
relative_path
return
path
if
pathname
.
relative?
pathname
.
relative_path_from
(
Pathname
.
new
(
root
))
end
def
model_valid?
def
model_valid?
!!
model
!!
model
end
end
...
@@ -115,4 +121,8 @@ class GitlabUploader < CarrierWave::Uploader::Base
...
@@ -115,4 +121,8 @@ class GitlabUploader < CarrierWave::Uploader::Base
# the cache directory.
# the cache directory.
File
.
join
(
work_dir
,
cache_id
,
version_name
.
to_s
,
for_file
)
File
.
join
(
work_dir
,
cache_id
,
version_name
.
to_s
,
for_file
)
end
end
def
pathname
@pathname
||=
Pathname
.
new
(
path
)
end
end
end
app/uploaders/job_artifact_uploader.rb
View file @
6f6c3c6f
...
@@ -9,6 +9,8 @@ class JobArtifactUploader < GitlabUploader
...
@@ -9,6 +9,8 @@ class JobArtifactUploader < GitlabUploader
storage_options
Gitlab
.
config
.
artifacts
storage_options
Gitlab
.
config
.
artifacts
alias_method
:upload
,
:model
def
cached_size
def
cached_size
return
model
.
size
if
model
.
size
.
present?
&&
!
model
.
file_changed?
return
model
.
size
if
model
.
size
.
present?
&&
!
model
.
file_changed?
...
...
app/uploaders/legacy_artifact_uploader.rb
View file @
6f6c3c6f
...
@@ -8,6 +8,8 @@ class LegacyArtifactUploader < GitlabUploader
...
@@ -8,6 +8,8 @@ class LegacyArtifactUploader < GitlabUploader
storage_options
Gitlab
.
config
.
artifacts
storage_options
Gitlab
.
config
.
artifacts
alias_method
:upload
,
:model
def
store_dir
def
store_dir
dynamic_segment
dynamic_segment
end
end
...
...
app/uploaders/lfs_object_uploader.rb
View file @
6f6c3c6f
...
@@ -6,6 +6,8 @@ class LfsObjectUploader < GitlabUploader
...
@@ -6,6 +6,8 @@ class LfsObjectUploader < GitlabUploader
storage_options
Gitlab
.
config
.
lfs
storage_options
Gitlab
.
config
.
lfs
alias_method
:upload
,
:model
def
filename
def
filename
model
.
oid
[
4
..-
1
]
model
.
oid
[
4
..-
1
]
end
end
...
...
app/uploaders/object_storage.rb
View file @
6f6c3c6f
...
@@ -137,6 +137,8 @@ module ObjectStorage
...
@@ -137,6 +137,8 @@ module ObjectStorage
included
do
|
base
|
included
do
|
base
|
base
.
include
(
ObjectStorage
)
base
.
include
(
ObjectStorage
)
include
::
EE
::
ObjectStorage
::
Concern
after
:migrate
,
:delete_migrated_file
after
:migrate
,
:delete_migrated_file
end
end
...
...
ee/app/models/ee/ci/build.rb
View file @
6f6c3c6f
...
@@ -81,6 +81,12 @@ module EE
...
@@ -81,6 +81,12 @@ module EE
has_artifact?
(
DAST_FILE
)
has_artifact?
(
DAST_FILE
)
end
end
def
log_geo_deleted_event
# It is not needed to generate a Geo deleted event
# since Legacy Artifacts are migrated to multi-build artifacts
# See https://gitlab.com/gitlab-org/gitlab-ce/issues/46652
end
private
private
def
has_artifact?
(
name
)
def
has_artifact?
(
name
)
...
...
ee/app/models/ee/ci/job_artifact.rb
View file @
6f6c3c6f
...
@@ -7,15 +7,13 @@ module EE
...
@@ -7,15 +7,13 @@ module EE
extend
ActiveSupport
::
Concern
extend
ActiveSupport
::
Concern
prepended
do
prepended
do
after_destroy
:log_geo_event
after_destroy
:log_geo_
deleted_
event
scope
:not_expired
,
->
{
where
(
'expire_at IS NULL OR expire_at > ?'
,
Time
.
current
)
}
scope
:not_expired
,
->
{
where
(
'expire_at IS NULL OR expire_at > ?'
,
Time
.
current
)
}
scope
:geo_syncable
,
->
{
with_files_stored_locally
.
not_expired
}
scope
:geo_syncable
,
->
{
with_files_stored_locally
.
not_expired
}
end
end
private
def
log_geo_deleted_event
def
log_geo_event
::
Geo
::
JobArtifactDeletedEventStore
.
new
(
self
).
create!
::
Geo
::
JobArtifactDeletedEventStore
.
new
(
self
).
create!
end
end
end
end
...
...
ee/app/models/ee/lfs_object.rb
View file @
6f6c3c6f
...
@@ -7,15 +7,13 @@ module EE
...
@@ -7,15 +7,13 @@ module EE
extend
ActiveSupport
::
Concern
extend
ActiveSupport
::
Concern
prepended
do
prepended
do
after_destroy
:log_geo_event
after_destroy
:log_geo_
deleted_
event
scope
:geo_syncable
,
->
{
with_files_stored_locally
}
scope
:geo_syncable
,
->
{
with_files_stored_locally
}
scope
:with_files_stored_remotely
,
->
{
where
(
file_store:
LfsObjectUploader
::
Store
::
REMOTE
)
}
scope
:with_files_stored_remotely
,
->
{
where
(
file_store:
LfsObjectUploader
::
Store
::
REMOTE
)
}
end
end
private
def
log_geo_deleted_event
def
log_geo_event
::
Geo
::
LfsObjectDeletedEventStore
.
new
(
self
).
create!
::
Geo
::
LfsObjectDeletedEventStore
.
new
(
self
).
create!
end
end
end
end
...
...
ee/app/models/ee/note.rb
View file @
6f6c3c6f
...
@@ -4,7 +4,7 @@ module EE
...
@@ -4,7 +4,7 @@ module EE
extend
::
Gitlab
::
Utils
::
Override
extend
::
Gitlab
::
Utils
::
Override
prepended
do
prepended
do
include
ObjectStorage
::
BackgroundMove
include
::
ObjectStorage
::
BackgroundMove
end
end
def
for_epic?
def
for_epic?
...
...
ee/app/models/ee/upload.rb
View file @
6f6c3c6f
...
@@ -7,14 +7,12 @@ module EE
...
@@ -7,14 +7,12 @@ module EE
extend
ActiveSupport
::
Concern
extend
ActiveSupport
::
Concern
prepended
do
prepended
do
after_destroy
:log_geo_event
after_destroy
:log_geo_
deleted_
event
scope
:geo_syncable
,
->
{
with_files_stored_locally
}
scope
:geo_syncable
,
->
{
with_files_stored_locally
}
end
end
private
def
log_geo_deleted_event
def
log_geo_event
::
Geo
::
UploadDeletedEventStore
.
new
(
self
).
create!
::
Geo
::
UploadDeletedEventStore
.
new
(
self
).
create!
end
end
end
end
...
...
ee/app/services/geo/file_download_service.rb
View file @
6f6c3c6f
# frozen_string_literal: true
module
Geo
module
Geo
# This class is responsible for:
# This class is responsible for:
# * Finding the appropriate Downloader class for a FileRegistry record
# * Finding the appropriate Downloader class for a FileRegistry record
...
@@ -50,7 +52,7 @@ module Geo
...
@@ -50,7 +52,7 @@ module Geo
# rubocop: disable CodeReuse/ActiveRecord
# rubocop: disable CodeReuse/ActiveRecord
def
update_registry
(
bytes_downloaded
,
mark_as_synced
:,
missing_on_primary:
false
)
def
update_registry
(
bytes_downloaded
,
mark_as_synced
:,
missing_on_primary:
false
)
registry
=
registry
=
if
object_type
.
to_sym
==
:job_artifact
if
job_artifact?
Geo
::
JobArtifactRegistry
.
find_or_initialize_by
(
artifact_id:
object_db_id
)
Geo
::
JobArtifactRegistry
.
find_or_initialize_by
(
artifact_id:
object_db_id
)
else
else
Geo
::
FileRegistry
.
find_or_initialize_by
(
Geo
::
FileRegistry
.
find_or_initialize_by
(
...
...
ee/app/services/geo/file_registry_removal_service.rb
View file @
6f6c3c6f
# frozen_string_literal: true
module
Geo
module
Geo
class
FileRegistryRemovalService
<
FileService
class
FileRegistryRemovalService
<
FileService
include
::
Gitlab
::
Utils
::
StrongMemoize
include
::
Gitlab
::
Utils
::
StrongMemoize
...
@@ -25,7 +27,7 @@ module Geo
...
@@ -25,7 +27,7 @@ module Geo
log_info
(
'Local file & registry removed'
)
log_info
(
'Local file & registry removed'
)
end
end
rescue
SystemCallError
rescue
SystemCallError
=>
e
log_error
(
'Could not remove file'
,
e
.
message
)
log_error
(
'Could not remove file'
,
e
.
message
)
raise
raise
end
end
...
@@ -35,7 +37,7 @@ module Geo
...
@@ -35,7 +37,7 @@ module Geo
# rubocop: disable CodeReuse/ActiveRecord
# rubocop: disable CodeReuse/ActiveRecord
def
file_registry
def
file_registry
strong_memoize
(
:file_registry
)
do
strong_memoize
(
:file_registry
)
do
if
object_type
.
to_sym
==
:job_artifact
if
job_artifact?
::
Geo
::
JobArtifactRegistry
.
find_by
(
artifact_id:
object_db_id
)
::
Geo
::
JobArtifactRegistry
.
find_by
(
artifact_id:
object_db_id
)
else
else
::
Geo
::
FileRegistry
.
find_by
(
file_type:
object_type
,
file_id:
object_db_id
)
::
Geo
::
FileRegistry
.
find_by
(
file_type:
object_type
,
file_id:
object_db_id
)
...
@@ -59,16 +61,15 @@ module Geo
...
@@ -59,16 +61,15 @@ module Geo
end
end
end
end
# rubocop: disable CodeReuse/ActiveRecord
def
file_uploader
def
file_uploader
strong_memoize
(
:file_uploader
)
do
strong_memoize
(
:file_uploader
)
do
case
object_type
.
to_s
case
object_type
when
'lfs'
when
:lfs
LfsObject
.
find
_by!
(
id:
object_db_id
).
file
LfsObject
.
find
(
object_db_id
).
file
when
'job_artifact'
when
:job_artifact
Ci
::
JobArtifact
.
find
_by!
(
id:
object_db_id
).
file
Ci
::
JobArtifact
.
find
(
object_db_id
).
file
when
*
Geo
::
FileService
::
DEFAULT_OBJECT_TYPES
when
*
Geo
::
FileService
::
DEFAULT_OBJECT_TYPES
Upload
.
find
_by!
(
id:
object_db_id
).
build_uploader
Upload
.
find
(
object_db_id
).
build_uploader
else
else
raise
NameError
,
"Unrecognized type:
#{
object_type
}
"
raise
NameError
,
"Unrecognized type:
#{
object_type
}
"
end
end
...
@@ -77,11 +78,6 @@ module Geo
...
@@ -77,11 +78,6 @@ module Geo
log_error
(
'Could not build uploader'
,
err
.
message
)
log_error
(
'Could not build uploader'
,
err
.
message
)
raise
raise
end
end
# rubocop: enable CodeReuse/ActiveRecord
def
upload?
Geo
::
FileService
::
DEFAULT_OBJECT_TYPES
.
include?
(
object_type
.
to_s
)
end
def
lease_key
def
lease_key
"file_registry_removal_service:
#{
object_type
}
:
#{
object_db_id
}
"
"file_registry_removal_service:
#{
object_type
}
:
#{
object_db_id
}
"
...
...
ee/app/services/geo/file_service.rb
View file @
6f6c3c6f
# frozen_string_literal: true
module
Geo
module
Geo
class
FileService
class
FileService
include
ExclusiveLeaseGuard
include
ExclusiveLeaseGuard
...
@@ -5,8 +7,8 @@ module Geo
...
@@ -5,8 +7,8 @@ module Geo
attr_reader
:object_type
,
:object_db_id
attr_reader
:object_type
,
:object_db_id
DEFAULT_OBJECT_TYPES
=
%
w
[attachment avatar file import_export namespace_file personal_file favicon]
.
freeze
DEFAULT_OBJECT_TYPES
=
%
i
[attachment avatar file import_export namespace_file personal_file favicon]
.
freeze
DEFAULT_SERVICE_TYPE
=
'file'
.
freez
e
DEFAULT_SERVICE_TYPE
=
:fil
e
def
initialize
(
object_type
,
object_db_id
)
def
initialize
(
object_type
,
object_db_id
)
@object_type
=
object_type
.
to_sym
@object_type
=
object_type
.
to_sym
...
@@ -19,9 +21,17 @@ module Geo
...
@@ -19,9 +21,17 @@ module Geo
private
private
def
upload?
DEFAULT_OBJECT_TYPES
.
include?
(
object_type
)
end
def
job_artifact?
object_type
==
:job_artifact
end
def
service_klass_name
def
service_klass_name
klass_name
=
klass_name
=
if
DEFAULT_OBJECT_TYPES
.
include?
(
object_type
.
to_s
)
if
upload?
DEFAULT_SERVICE_TYPE
DEFAULT_SERVICE_TYPE
else
else
object_type
object_type
...
@@ -33,7 +43,7 @@ module Geo
...
@@ -33,7 +43,7 @@ module Geo
def
base_log_data
(
message
)
def
base_log_data
(
message
)
{
{
class:
self
.
class
.
name
,
class:
self
.
class
.
name
,
object_type:
object_type
.
to_s
,
object_type:
object_type
,
object_db_id:
object_db_id
,
object_db_id:
object_db_id
,
message:
message
message:
message
}
}
...
...
ee/app/services/geo/job_artifact_deleted_event_store.rb
View file @
6f6c3c6f
...
@@ -10,13 +10,6 @@ module Geo
...
@@ -10,13 +10,6 @@ module Geo
@job_artifact
=
job_artifact
@job_artifact
=
job_artifact
end
end
override
:create!
def
create!
return
unless
job_artifact
.
local_store?
super
end
private
private
def
build_event
def
build_event
...
@@ -26,14 +19,8 @@ module Geo
...
@@ -26,14 +19,8 @@ module Geo
)
)
end
end
def
local_store_path
Pathname
.
new
(
JobArtifactUploader
.
root
)
end
def
relative_file_path
def
relative_file_path
return
unless
job_artifact
.
file
.
present?
job_artifact
.
file
.
relative_path
if
job_artifact
.
file
.
present?
Pathname
.
new
(
job_artifact
.
file
.
path
).
relative_path_from
(
local_store_path
)
end
end
# This is called by ProjectLogHelpers to build json log with context info
# This is called by ProjectLogHelpers to build json log with context info
...
...
ee/app/services/geo/lfs_object_deleted_event_store.rb
View file @
6f6c3c6f
...
@@ -10,13 +10,6 @@ module Geo
...
@@ -10,13 +10,6 @@ module Geo
@lfs_object
=
lfs_object
@lfs_object
=
lfs_object
end
end
override
:create!
def
create!
return
unless
lfs_object
.
local_store?
super
end
private
private
def
build_event
def
build_event
...
@@ -27,14 +20,8 @@ module Geo
...
@@ -27,14 +20,8 @@ module Geo
)
)
end
end
def
local_store_path
Pathname
.
new
(
LfsObjectUploader
.
root
)
end
def
relative_file_path
def
relative_file_path
return
unless
lfs_object
.
file
.
present?
lfs_object
.
file
.
relative_path
if
lfs_object
.
file
.
present?
Pathname
.
new
(
lfs_object
.
file
.
path
).
relative_path_from
(
local_store_path
)
end
end
# This is called by ProjectLogHelpers to build json log with context info
# This is called by ProjectLogHelpers to build json log with context info
...
...
ee/app/services/geo/upload_deleted_event_store.rb
View file @
6f6c3c6f
...
@@ -10,13 +10,6 @@ module Geo
...
@@ -10,13 +10,6 @@ module Geo
@upload
=
upload
@upload
=
upload
end
end
override
:create!
def
create!
return
unless
upload
.
local?
super
end
private
private
def
build_event
def
build_event
...
...
ee/app/uploaders/ee/object_storage.rb
0 → 100644
View file @
6f6c3c6f
module
EE
module
ObjectStorage
module
Concern
extend
ActiveSupport
::
Concern
included
do
after
:migrate
,
:log_geo_deleted_event
end
private
def
log_geo_deleted_event
(
_migrated_file
)
upload
.
log_geo_deleted_event
end
end
end
end
ee/changelogs/unreleased/tc-geo-log-object-storage.yml
0 → 100644
View file @
6f6c3c6f
---
title
:
Use Geo log to remove files when migrated to object storage
merge_request
:
5966
author
:
type
:
added
ee/lib/gitlab/geo/log_cursor/events/job_artifact_deleted_event.rb
View file @
6f6c3c6f
...
@@ -6,42 +6,20 @@ module Gitlab
...
@@ -6,42 +6,20 @@ module Gitlab
include
BaseEvent
include
BaseEvent
def
process
def
process
return
unless
file_registry_job_artifacts
.
any?
# avoid race condition
# Must always schedule, regardless of shard health
job_id
=
::
Geo
::
FileRegistryRemovalWorker
.
perform_async
(
:job_artifact
,
event
.
job_artifact_id
)
# delete synchronously to ensure consistency
log_event
(
job_id
)
if
File
.
file?
(
file_path
)
&&
!
delete_file
(
file_path
)
return
# do not delete file from registry if deletion failed
end
log_event
file_registry_job_artifacts
.
delete_all
end
end
private
private
# rubocop: disable CodeReuse/ActiveRecord
def
log_event
(
job_id
)
def
file_registry_job_artifacts
@file_registry_job_artifacts
||=
::
Geo
::
JobArtifactRegistry
.
where
(
artifact_id:
event
.
job_artifact_id
)
end
# rubocop: enable CodeReuse/ActiveRecord
def
file_path
@file_path
||=
File
.
join
(
::
JobArtifactUploader
.
root
,
event
.
file_path
)
end
def
log_event
logger
.
event_info
(
logger
.
event_info
(
created_at
,
created_at
,
'Delete
d job artifact
'
,
'Delete
job artifact scheduled
'
,
file_id:
event
.
job_artifact_id
,
file_id:
event
.
job_artifact_id
,
file_path:
file_path
)
file_path:
event
.
file_path
,
end
job_id:
job_id
)
def
delete_file
(
path
)
File
.
delete
(
path
)
rescue
=>
ex
logger
.
error
(
"Failed to remove file"
,
exception:
ex
.
class
.
name
,
details:
ex
.
message
,
filename:
path
)
false
end
end
end
end
end
end
...
...
ee/lib/gitlab/geo/log_cursor/events/lfs_object_deleted_event.rb
View file @
6f6c3c6f
...
@@ -5,28 +5,21 @@ module Gitlab
...
@@ -5,28 +5,21 @@ module Gitlab
class
LfsObjectDeletedEvent
class
LfsObjectDeletedEvent
include
BaseEvent
include
BaseEvent
# rubocop: disable CodeReuse/ActiveRecord
def
process
def
process
# Must always schedule, regardless of shard health
# Must always schedule, regardless of shard health
job_id
=
::
Geo
::
FileRe
movalWorker
.
perform_async
(
file_path
)
job_id
=
::
Geo
::
FileRe
gistryRemovalWorker
.
perform_async
(
:lfs
,
event
.
lfs_object_id
)
log_event
(
job_id
)
log_event
(
job_id
)
::
Geo
::
FileRegistry
.
lfs_objects
.
where
(
file_id:
event
.
lfs_object_id
).
delete_all
end
end
# rubocop: enable CodeReuse/ActiveRecord
private
private
def
file_path
@file_path
||=
File
.
join
(
LfsObjectUploader
.
root
,
event
.
file_path
)
end
def
log_event
(
job_id
)
def
log_event
(
job_id
)
logger
.
event_info
(
logger
.
event_info
(
created_at
,
created_at
,
'Delete
d LFS object
'
,
'Delete
LFS object scheduled
'
,
oid:
event
.
oid
,
oid:
event
.
oid
,
file_id:
event
.
lfs_object_id
,
file_id:
event
.
lfs_object_id
,
file_path:
file_path
,
file_path:
event
.
file_path
,
job_id:
job_id
)
job_id:
job_id
)
end
end
end
end
...
...
ee/lib/gitlab/geo/log_cursor/events/upload_deleted_event.rb
View file @
6f6c3c6f
...
@@ -5,24 +5,21 @@ module Gitlab
...
@@ -5,24 +5,21 @@ module Gitlab
class
UploadDeletedEvent
class
UploadDeletedEvent
include
BaseEvent
include
BaseEvent
# rubocop: disable CodeReuse/ActiveRecord
def
process
def
process
log_event
job_id
=
::
Geo
::
FileRegistryRemovalWorker
.
perform_async
(
event
.
upload_type
,
event
.
upload_id
)
::
Geo
::
FileRegistry
.
where
(
file_id:
event
.
upload_id
,
file_type:
event
.
upload_type
).
delete_all
log_event
(
job_id
)
end
end
# rubocop: enable CodeReuse/ActiveRecord
private
def
log_event
(
job_id
)
def
log_event
logger
.
event_info
(
logger
.
event_info
(
created_at
,
created_at
,
'Delete
d upload file
'
,
'Delete
upload file scheduled
'
,
upload_id:
event
.
upload_id
,
upload_id:
event
.
upload_id
,
upload_type:
event
.
upload_type
,
upload_type:
event
.
upload_type
,
file_path:
event
.
file_path
,
file_path:
event
.
file_path
,
model_id:
event
.
model_id
,
model_id:
event
.
model_id
,
model_type:
event
.
model_type
)
model_type:
event
.
model_type
,
job_id:
job_id
)
end
end
end
end
end
end
...
...
ee/spec/lib/gitlab/geo/log_cursor/events/job_artifact_deleted_event_spec.rb
View file @
6f6c3c6f
...
@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, :
...
@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, :
subject
{
described_class
.
new
(
job_artifact_deleted_event
,
Time
.
now
,
logger
)
}
subject
{
described_class
.
new
(
job_artifact_deleted_event
,
Time
.
now
,
logger
)
}
around
do
|
example
|
around
do
|
example
|
Sidekiq
::
Testing
.
fak
e!
{
example
.
run
}
Sidekiq
::
Testing
.
inlin
e!
{
example
.
run
}
end
end
describe
'#process'
do
describe
'#process'
do
...
@@ -32,11 +32,14 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, :
...
@@ -32,11 +32,14 @@ describe Gitlab::Geo::LogCursor::Events::JobArtifactDeletedEvent, :postgresql, :
context
'when the delete fails'
do
context
'when the delete fails'
do
before
do
before
do
expect
(
File
).
to
receive
(
:delete
).
with
(
job_artifact
.
file
.
path
).
and_raise
(
"Cannot delete"
)
allow
(
File
).
to
receive
(
:unlink
).
and_call_original
allow
(
File
).
to
receive
(
:unlink
).
with
(
job_artifact
.
file
.
path
).
and_raise
(
SystemCallError
,
"Cannot delete"
)
end
end
it
'does not remove the tracking database entry'
do
it
'does not remove the tracking database entry'
do
expect
{
subject
.
process
}.
not_to
change
(
Geo
::
JobArtifactRegistry
,
:count
)
expect
do
expect
{
subject
.
process
}.
to
raise_error
(
SystemCallError
)
end
.
not_to
change
(
Geo
::
JobArtifactRegistry
,
:count
)
end
end
end
end
end
end
...
...
ee/spec/lib/gitlab/geo/log_cursor/events/lfs_object_deleted_event_spec.rb
View file @
6f6c3c6f
...
@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl
...
@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl
subject
{
described_class
.
new
(
lfs_object_deleted_event
,
Time
.
now
,
logger
)
}
subject
{
described_class
.
new
(
lfs_object_deleted_event
,
Time
.
now
,
logger
)
}
around
do
|
example
|
around
do
|
example
|
Sidekiq
::
Testing
.
fak
e!
{
example
.
run
}
Sidekiq
::
Testing
.
inlin
e!
{
example
.
run
}
end
end
describe
'#process'
do
describe
'#process'
do
...
@@ -24,10 +24,8 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl
...
@@ -24,10 +24,8 @@ describe Gitlab::Geo::LogCursor::Events::LfsObjectDeletedEvent, :postgresql, :cl
expect
{
subject
.
process
}.
to
change
(
Geo
::
FileRegistry
.
lfs_objects
,
:count
).
by
(
-
1
)
expect
{
subject
.
process
}.
to
change
(
Geo
::
FileRegistry
.
lfs_objects
,
:count
).
by
(
-
1
)
end
end
it
'schedules a Geo::FileRemovalWorker job'
do
it
'schedules a Geo::FileRegistryRemovalWorker job'
do
file_path
=
File
.
join
(
LfsObjectUploader
.
root
,
lfs_object_deleted_event
.
file_path
)
expect
(
::
Geo
::
FileRegistryRemovalWorker
).
to
receive
(
:perform_async
).
with
(
:lfs
,
lfs_object_deleted_event
.
lfs_object_id
)
expect
(
::
Geo
::
FileRemovalWorker
).
to
receive
(
:perform_async
).
with
(
file_path
)
subject
.
process
subject
.
process
end
end
...
...
ee/spec/lib/gitlab/geo/log_cursor/events/upload_deleted_event_spec.rb
View file @
6f6c3c6f
...
@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::UploadDeletedEvent, :postgresql, :clean
...
@@ -10,7 +10,7 @@ describe Gitlab::Geo::LogCursor::Events::UploadDeletedEvent, :postgresql, :clean
subject
{
described_class
.
new
(
upload_deleted_event
,
Time
.
now
,
logger
)
}
subject
{
described_class
.
new
(
upload_deleted_event
,
Time
.
now
,
logger
)
}
around
do
|
example
|
around
do
|
example
|
Sidekiq
::
Testing
.
fak
e!
{
example
.
run
}
Sidekiq
::
Testing
.
inlin
e!
{
example
.
run
}
end
end
describe
'#process'
do
describe
'#process'
do
...
...
ee/spec/models/ee/appearance_spec.rb
View file @
6f6c3c6f
require
'spec_helper'
require
'spec_helper'
describe
Appearance
do
describe
Appearance
do
include
::
EE
::
GeoHelpers
subject
{
build
(
:appearance
)
}
subject
{
build
(
:appearance
)
}
describe
'validations'
do
describe
'validations'
do
...
@@ -17,4 +19,42 @@ describe Appearance do
...
@@ -17,4 +19,42 @@ describe Appearance do
it
{
is_expected
.
to
allow_value
(
hex
).
for
(
:message_font_color
)
}
it
{
is_expected
.
to
allow_value
(
hex
).
for
(
:message_font_color
)
}
it
{
is_expected
.
not_to
allow_value
(
'000'
).
for
(
:message_font_color
)
}
it
{
is_expected
.
not_to
allow_value
(
'000'
).
for
(
:message_font_color
)
}
end
end
context
'object storage with background upload'
do
context
'when running in a Geo primary node'
do
set
(
:primary
)
{
create
(
:geo_node
,
:primary
)
}
set
(
:secondary
)
{
create
(
:geo_node
)
}
before
do
stub_current_geo_node
(
primary
)
stub_uploads_object_storage
(
AttachmentUploader
,
background_upload:
true
)
end
it
'creates a Geo deleted log event for logo'
do
Sidekiq
::
Testing
.
inline!
do
expect
do
create
(
:appearance
,
:with_logo
)
end
.
to
change
(
Geo
::
UploadDeletedEvent
,
:count
).
by
(
1
)
end
end
it
'creates a Geo deleted log event for header logo'
do
Sidekiq
::
Testing
.
inline!
do
expect
do
create
(
:appearance
,
:with_header_logo
)
end
.
to
change
(
Geo
::
UploadDeletedEvent
,
:count
).
by
(
1
)
end
end
it
'creates only a Geo deleted log event for the migrated header logo'
do
Sidekiq
::
Testing
.
inline!
do
appearance
=
create
(
:appearance
,
:with_header_logo
,
:with_logo
)
expect
do
appearance
.
update
(
header_logo:
fixture_file_upload
(
'spec/fixtures/rails_sample.jpg'
))
end
.
to
change
(
Geo
::
UploadDeletedEvent
,
:count
).
by
(
1
)
end
end
end
end
end
end
ee/spec/models/ee/note_spec.rb
0 → 100644
View file @
6f6c3c6f
require
'spec_helper'
describe
Note
do
include
::
EE
::
GeoHelpers
context
'object storage with background upload'
do
before
do
stub_uploads_object_storage
(
AttachmentUploader
,
background_upload:
true
)
end
context
'when running in a Geo primary node'
do
set
(
:primary
)
{
create
(
:geo_node
,
:primary
)
}
set
(
:secondary
)
{
create
(
:geo_node
)
}
before
do
stub_current_geo_node
(
primary
)
end
it
'creates a Geo deleted log event for attachment'
do
Sidekiq
::
Testing
.
inline!
do
expect
do
create
(
:note
,
:with_attachment
)
end
.
to
change
(
Geo
::
UploadDeletedEvent
,
:count
).
by
(
1
)
end
end
end
end
end
ee/spec/services/geo/file_registry_removal_service_spec.rb
View file @
6f6c3c6f
...
@@ -129,7 +129,7 @@ describe Geo::FileRegistryRemovalService do
...
@@ -129,7 +129,7 @@ describe Geo::FileRegistryRemovalService do
end
end
end
end
context
'with file'
do
# TODO
context
'with file'
do
let!
(
:upload
)
{
create
(
:user
,
:with_avatar
).
avatar
.
upload
}
let!
(
:upload
)
{
create
(
:user
,
:with_avatar
).
avatar
.
upload
}
let!
(
:file_registry
)
{
create
(
:geo_file_registry
,
:avatar
,
file_id:
upload
.
id
)
}
let!
(
:file_registry
)
{
create
(
:geo_file_registry
,
:avatar
,
file_id:
upload
.
id
)
}
let!
(
:file_path
)
{
upload
.
build_uploader
.
file
.
path
}
let!
(
:file_path
)
{
upload
.
build_uploader
.
file
.
path
}
...
...
ee/spec/services/geo/hashed_storage_attachments_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -18,7 +18,7 @@ describe Geo::HashedStorageAttachmentsEventStore do
...
@@ -18,7 +18,7 @@ describe Geo::HashedStorageAttachmentsEventStore do
TestEnv
.
clean_test_path
TestEnv
.
clean_test_path
end
end
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
HashedStorageAttachmentsEvent
it_behaves_like
'a Geo event store'
,
Geo
::
HashedStorageAttachmentsEvent
context
'when running on a primary node'
do
context
'when running on a primary node'
do
...
...
ee/spec/services/geo/hashed_storage_migrated_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -17,7 +17,7 @@ describe Geo::HashedStorageMigratedEventStore do
...
@@ -17,7 +17,7 @@ describe Geo::HashedStorageMigratedEventStore do
TestEnv
.
clean_test_path
TestEnv
.
clean_test_path
end
end
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
HashedStorageMigratedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
HashedStorageMigratedEvent
context
'when running on a primary node'
do
context
'when running on a primary node'
do
...
...
ee/spec/services/geo/job_artifact_deleted_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -11,21 +11,17 @@ describe Geo::JobArtifactDeletedEventStore do
...
@@ -11,21 +11,17 @@ describe Geo::JobArtifactDeletedEventStore do
subject
{
described_class
.
new
(
job_artifact
)
}
subject
{
described_class
.
new
(
job_artifact
)
}
describe
'#create'
do
describe
'#create!'
do
it_behaves_like
'a Geo event store'
,
Geo
::
JobArtifactDeletedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
JobArtifactDeletedEvent
do
let
(
:file_subject
)
{
job_artifact
}
end
context
'when running on a primary node'
do
context
'when running on a primary node'
do
before
do
before
do
stub_primary_node
stub_primary_node
end
end
it
'does not create an event when LFS object is not on a local store'
do
it
'tracks artifact attributes'
do
allow
(
job_artifact
).
to
receive
(
:local_store?
).
and_return
(
false
)
expect
{
subject
.
create!
}.
not_to
change
(
Geo
::
JobArtifactDeletedEvent
,
:count
)
end
it
'tracks LFS object attributes'
do
subject
.
create!
subject
.
create!
expect
(
Geo
::
JobArtifactDeletedEvent
.
last
).
to
have_attributes
(
expect
(
Geo
::
JobArtifactDeletedEvent
.
last
).
to
have_attributes
(
...
...
ee/spec/services/geo/lfs_object_deleted_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -11,20 +11,16 @@ describe Geo::LfsObjectDeletedEventStore do
...
@@ -11,20 +11,16 @@ describe Geo::LfsObjectDeletedEventStore do
subject
{
described_class
.
new
(
lfs_object
)
}
subject
{
described_class
.
new
(
lfs_object
)
}
describe
'#create'
do
describe
'#create!'
do
it_behaves_like
'a Geo event store'
,
Geo
::
LfsObjectDeletedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
LfsObjectDeletedEvent
do
let
(
:file_subject
)
{
lfs_object
}
end
context
'when running on a primary node'
do
context
'when running on a primary node'
do
before
do
before
do
stub_primary_node
stub_primary_node
end
end
it
'does not create an event when LFS object is not on a local store'
do
allow
(
lfs_object
).
to
receive
(
:local_store?
).
and_return
(
false
)
expect
{
subject
.
create!
}.
not_to
change
(
Geo
::
LfsObjectDeletedEvent
,
:count
)
end
it
'tracks LFS object attributes'
do
it
'tracks LFS object attributes'
do
subject
.
create!
subject
.
create!
...
...
ee/spec/services/geo/repositories_changed_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -9,7 +9,7 @@ describe Geo::RepositoriesChangedEventStore do
...
@@ -9,7 +9,7 @@ describe Geo::RepositoriesChangedEventStore do
subject
{
described_class
.
new
(
geo_node
)
}
subject
{
described_class
.
new
(
geo_node
)
}
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoriesChangedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoriesChangedEvent
end
end
end
end
ee/spec/services/geo/repository_created_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -10,7 +10,7 @@ describe Geo::RepositoryCreatedEventStore do
...
@@ -10,7 +10,7 @@ describe Geo::RepositoryCreatedEventStore do
subject
{
described_class
.
new
(
project
)
}
subject
{
described_class
.
new
(
project
)
}
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryCreatedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryCreatedEvent
context
'running on a primary node'
do
context
'running on a primary node'
do
...
...
ee/spec/services/geo/repository_deleted_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -16,7 +16,7 @@ describe Geo::RepositoryDeletedEventStore do
...
@@ -16,7 +16,7 @@ describe Geo::RepositoryDeletedEventStore do
subject
{
described_class
.
new
(
project
,
repo_path:
repo_path
,
wiki_path:
wiki_path
)
}
subject
{
described_class
.
new
(
project
,
repo_path:
repo_path
,
wiki_path:
wiki_path
)
}
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryDeletedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryDeletedEvent
context
'when running on a primary node'
do
context
'when running on a primary node'
do
...
...
ee/spec/services/geo/repository_renamed_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -13,7 +13,7 @@ describe Geo::RepositoryRenamedEventStore do
...
@@ -13,7 +13,7 @@ describe Geo::RepositoryRenamedEventStore do
subject
{
described_class
.
new
(
project
,
old_path:
old_path
,
old_path_with_namespace:
old_path_with_namespace
)
}
subject
{
described_class
.
new
(
project
,
old_path:
old_path
,
old_path_with_namespace:
old_path_with_namespace
)
}
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryRenamedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryRenamedEvent
context
'when running on a primary node'
do
context
'when running on a primary node'
do
...
...
ee/spec/services/geo/repository_updated_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -20,7 +20,7 @@ describe Geo::RepositoryUpdatedEventStore do
...
@@ -20,7 +20,7 @@ describe Geo::RepositoryUpdatedEventStore do
subject
{
described_class
.
new
(
project
,
refs:
refs
,
changes:
changes
)
}
subject
{
described_class
.
new
(
project
,
refs:
refs
,
changes:
changes
)
}
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryUpdatedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
RepositoryUpdatedEvent
context
'when running on a primary node'
do
context
'when running on a primary node'
do
...
...
ee/spec/services/geo/reset_checksum_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -10,7 +10,7 @@ describe Geo::ResetChecksumEventStore do
...
@@ -10,7 +10,7 @@ describe Geo::ResetChecksumEventStore do
subject
{
described_class
.
new
(
project
)
}
subject
{
described_class
.
new
(
project
)
}
describe
'#create'
do
describe
'#create
!
'
do
it_behaves_like
'a Geo event store'
,
Geo
::
ResetChecksumEvent
it_behaves_like
'a Geo event store'
,
Geo
::
ResetChecksumEvent
context
'when running on a primary node'
do
context
'when running on a primary node'
do
...
...
ee/spec/services/geo/upload_deleted_event_store_spec.rb
View file @
6f6c3c6f
...
@@ -11,20 +11,16 @@ describe Geo::UploadDeletedEventStore do
...
@@ -11,20 +11,16 @@ describe Geo::UploadDeletedEventStore do
subject
{
described_class
.
new
(
upload
)
}
subject
{
described_class
.
new
(
upload
)
}
describe
'#create'
do
describe
'#create!'
do
it_behaves_like
'a Geo event store'
,
Geo
::
UploadDeletedEvent
it_behaves_like
'a Geo event store'
,
Geo
::
UploadDeletedEvent
do
let
(
:file_subject
)
{
upload
}
end
context
'when running on a primary node'
do
context
'when running on a primary node'
do
before
do
before
do
stub_primary_node
stub_primary_node
end
end
it
'does not create an event when the upload does not use local storage'
do
allow
(
upload
).
to
receive
(
:local?
).
and_return
(
false
)
expect
{
subject
.
create!
}.
not_to
change
(
Geo
::
UploadDeletedEvent
,
:count
)
end
it
'tracks upload attributes'
do
it
'tracks upload attributes'
do
subject
.
create!
subject
.
create!
...
...
ee/spec/support/shared_examples/geo_event_store_shared_examples.rb
View file @
6f6c3c6f
...
@@ -25,5 +25,17 @@ shared_examples_for 'a Geo event store' do |event_class|
...
@@ -25,5 +25,17 @@ shared_examples_for 'a Geo event store' do |event_class|
it
'creates an event'
do
it
'creates an event'
do
expect
{
subject
.
create!
}.
to
change
(
event_class
,
:count
).
by
(
1
)
expect
{
subject
.
create!
}.
to
change
(
event_class
,
:count
).
by
(
1
)
end
end
context
'when file subject is not on local store'
do
before
do
skip
'No file subject defined, skipping'
unless
defined?
(
file_subject
)
allow
(
file_subject
).
to
receive
(
:local?
).
and_return
(
false
)
end
it
'creates an event'
do
expect
{
subject
.
create!
}.
to
change
(
event_class
,
:count
).
by
(
1
)
end
end
end
end
end
end
spec/support/shared_examples/uploaders/gitlab_uploader_shared_examples.rb
View file @
6f6c3c6f
...
@@ -33,6 +33,14 @@ shared_examples "builds correct paths" do |**patterns|
...
@@ -33,6 +33,14 @@ shared_examples "builds correct paths" do |**patterns|
it_behaves_like
"matches the method pattern"
,
:upload_path
it_behaves_like
"matches the method pattern"
,
:upload_path
end
end
describe
"#relative_path"
do
it
'is relative'
do
skip
'Path not set, skipping.'
unless
subject
.
path
expect
(
Pathname
.
new
(
subject
.
relative_path
)).
to
be_relative
end
end
describe
".absolute_path"
do
describe
".absolute_path"
do
it_behaves_like
"matches the method pattern"
,
:absolute_path
do
it_behaves_like
"matches the method pattern"
,
:absolute_path
do
let
(
:target
)
{
subject
.
class
}
let
(
:target
)
{
subject
.
class
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment