Commit c7837b50 authored by Rémy Coutable's avatar Rémy Coutable

Merge branch '2902-standalone-ee-dir' into 'master'

Move EE-specific files to a standalone directory

Closes #2902

See merge request !2483
parents 6c1c5b19 0be3a416
/* global ListIssue */ /* global ListIssue */
import Vue from 'vue'; import Vue from 'vue';
import queryData from '../../utils/query_data'; import queryData from '~/boards/utils/query_data';
import loadingIcon from '../../../vue_shared/components/loading_icon.vue'; import loadingIcon from '~/vue_shared/components/loading_icon.vue';
import './header'; import './header';
import './list'; import './list';
import './footer'; import './footer';
......
<script> <script>
import GfmAutoComplete from '~/gfm_auto_complete'; import GfmAutoComplete from '~/gfm_auto_complete';
import loadingIcon from '~/vue_shared/components/loading_icon.vue';
import eventHub from '../event_hub'; import eventHub from '../event_hub';
import issueToken from './issue_token.vue'; import issueToken from './issue_token.vue';
import loadingIcon from '../../../vue_shared/components/loading_icon.vue';
export default { export default {
name: 'AddIssuableForm', name: 'AddIssuableForm',
......
<script> <script>
import loadingIcon from '~/vue_shared/components/loading_icon.vue';
import tooltip from '~/vue_shared/directives/tooltip';
import eventHub from '../event_hub'; import eventHub from '../event_hub';
import loadingIcon from '../../../vue_shared/components/loading_icon.vue';
import tooltip from '../../../vue_shared/directives/tooltip';
import issueToken from './issue_token.vue'; import issueToken from './issue_token.vue';
import addIssuableForm from './add_issuable_form.vue'; import addIssuableForm from './add_issuable_form.vue';
......
<script> <script>
import loadingIcon from '~/vue_shared/components/loading_icon.vue';
import linkedPipelinesColumn from './linked_pipelines_column.vue'; import linkedPipelinesColumn from './linked_pipelines_column.vue';
import stageColumnComponent from './stage_column_component.vue'; import stageColumnComponent from './stage_column_component.vue';
import loadingIcon from '../../../vue_shared/components/loading_icon.vue';
export default { export default {
props: { props: {
......
/** /**
* This file is the centerpiece of an attempt to reduce potential conflicts * This file is the centerpiece of an attempt to reduce potential conflicts
* between the CE and EE versions of the MR widget. EE additions to the MR widget should * between the CE and EE versions of the MR widget. EE additions to the MR widget should
* be contained in the ./vue_merge_request_widget/ee directory, and should **extend** * be contained in the ee/vue_merge_request_widget directory, and should **extend**
* rather than mutate CE MR Widget code. * rather than mutate CE MR Widget code.
* *
* This file should be the only source of conflicts between EE and CE. EE-only components should * This file should be the only source of conflicts between EE and CE. EE-only components should
...@@ -26,7 +26,7 @@ export { default as ConflictsState } from './components/states/mr_widget_conflic ...@@ -26,7 +26,7 @@ export { default as ConflictsState } from './components/states/mr_widget_conflic
export { default as NothingToMergeState } from './components/states/mr_widget_nothing_to_merge'; export { default as NothingToMergeState } from './components/states/mr_widget_nothing_to_merge';
export { default as MissingBranchState } from './components/states/mr_widget_missing_branch'; export { default as MissingBranchState } from './components/states/mr_widget_missing_branch';
export { default as NotAllowedState } from './components/states/mr_widget_not_allowed'; export { default as NotAllowedState } from './components/states/mr_widget_not_allowed';
export { default as ReadyToMergeState } from './ee/components/states/mr_widget_ready_to_merge'; export { default as ReadyToMergeState } from 'ee/vue_merge_request_widget/components/states/mr_widget_ready_to_merge';
export { default as SHAMismatchState } from './components/states/mr_widget_sha_mismatch'; export { default as SHAMismatchState } from './components/states/mr_widget_sha_mismatch';
export { default as UnresolvedDiscussionsState } from './components/states/mr_widget_unresolved_discussions'; export { default as UnresolvedDiscussionsState } from './components/states/mr_widget_unresolved_discussions';
export { default as PipelineBlockedState } from './components/states/mr_widget_pipeline_blocked'; export { default as PipelineBlockedState } from './components/states/mr_widget_pipeline_blocked';
...@@ -34,11 +34,11 @@ export { default as PipelineFailedState } from './components/states/mr_widget_pi ...@@ -34,11 +34,11 @@ export { default as PipelineFailedState } from './components/states/mr_widget_pi
export { default as MergeWhenPipelineSucceedsState } from './components/states/mr_widget_merge_when_pipeline_succeeds'; export { default as MergeWhenPipelineSucceedsState } from './components/states/mr_widget_merge_when_pipeline_succeeds';
export { default as AutoMergeFailed } from './components/states/mr_widget_auto_merge_failed'; export { default as AutoMergeFailed } from './components/states/mr_widget_auto_merge_failed';
export { default as CheckingState } from './components/states/mr_widget_checking'; export { default as CheckingState } from './components/states/mr_widget_checking';
export { default as MRWidgetStore } from './ee/stores/mr_widget_store'; export { default as MRWidgetStore } from 'ee/vue_merge_request_widget/stores/mr_widget_store';
export { default as MRWidgetService } from './ee/services/mr_widget_service'; export { default as MRWidgetService } from 'ee/vue_merge_request_widget/services/mr_widget_service';
export { default as eventHub } from './event_hub'; export { default as eventHub } from './event_hub';
export { default as getStateKey } from './ee/stores/get_state_key'; export { default as getStateKey } from 'ee/vue_merge_request_widget/stores/get_state_key';
export { default as mrWidgetOptions } from './ee/mr_widget_options'; export { default as mrWidgetOptions } from 'ee/vue_merge_request_widget/mr_widget_options';
export { default as stateMaps } from './ee/stores/state_maps'; export { default as stateMaps } from 'ee/vue_merge_request_widget/stores/state_maps';
export { default as SquashBeforeMerge } from './ee/components/states/mr_widget_squash_before_merge'; export { default as SquashBeforeMerge } from 'ee/vue_merge_request_widget/components/states/mr_widget_squash_before_merge';
export { default as notify } from '../lib/utils/notify'; export { default as notify } from '../lib/utils/notify';
# Shorter routing method for some project items # Shorter routing method for some project items
module GitlabRoutingHelper module GitlabRoutingHelper
include EE::GitlabRoutingHelper
extend ActiveSupport::Concern extend ActiveSupport::Concern
included do included do
......
module IssuesHelper module IssuesHelper
include EE::IssuesHelper
def issue_css_classes(issue) def issue_css_classes(issue)
classes = "issue" classes = "issue"
classes << " closed" if issue.closed? classes << " closed" if issue.closed?
......
module NamespacesHelper module NamespacesHelper
include EE::NamespaceHelper
def namespace_id_from(params) def namespace_id_from(params)
params.dig(:project, :namespace_id) || params[:namespace_id] params.dig(:project, :namespace_id) || params[:namespace_id]
end end
......
...@@ -25,19 +25,33 @@ module Gitlab ...@@ -25,19 +25,33 @@ module Gitlab
# https://github.com/rails/rails/blob/v4.2.6/railties/lib/rails/engine.rb#L687 # https://github.com/rails/rails/blob/v4.2.6/railties/lib/rails/engine.rb#L687
# This is a nice reference article on autoloading/eager loading: # This is a nice reference article on autoloading/eager loading:
# http://blog.arkency.com/2014/11/dont-forget-about-eager-load-when-extending-autoload # http://blog.arkency.com/2014/11/dont-forget-about-eager-load-when-extending-autoload
config.eager_load_paths.push(*%W(#{config.root}/lib config.eager_load_paths.push(*%W[#{config.root}/lib
#{config.root}/app/models/hooks #{config.root}/app/models/hooks
#{config.root}/app/models/members #{config.root}/app/models/members
#{config.root}/app/models/project_services #{config.root}/app/models/project_services
#{config.root}/app/workers/concerns #{config.root}/app/workers/concerns
#{config.root}/app/services/concerns #{config.root}/app/services/concerns
#{config.root}/app/uploaders/concerns #{config.root}/app/uploaders/concerns
#{config.root}/app/finders/concerns)) #{config.root}/app/finders/concerns])
config.generators.templates.push("#{config.root}/generator_templates") config.generators.templates.push("#{config.root}/generator_templates")
# EE specific paths. # EE specific paths.
config.eager_load_paths.push("#{config.root}/app/workers/concerns") config.eager_load_paths.push(*%W[
#{config.root}/ee/lib
#{config.root}/ee/app/controllers
#{config.root}/ee/app/helpers
#{config.root}/ee/app/mailers
#{config.root}/ee/app/models
#{config.root}/ee/app/models/concerns
#{config.root}/ee/app/policies
#{config.root}/ee/app/services
#{config.root}/ee/app/workers
])
config.paths['app/views'].push(*%W[
#{config.root}/ee/app/views
])
# Only load the plugins named here, in the order given (default is alphabetical). # Only load the plugins named here, in the order given (default is alphabetical).
# :all can be used as a placeholder for all plugins not explicitly named. # :all can be used as a placeholder for all plugins not explicitly named.
......
require_dependency Rails.root.join('lib/gitlab') # Load Gitlab as soon as possible require_dependency Rails.root.join('lib/gitlab') # Load Gitlab as soon as possible
require_dependency Rails.root.join('lib/ee') # Load EE as soon as possible
class Settings < Settingslogic class Settings < Settingslogic
source ENV.fetch('GITLAB_CONFIG') { "#{Rails.root}/config/gitlab.yml" } source ENV.fetch('GITLAB_CONFIG') { "#{Rails.root}/config/gitlab.yml" }
......
...@@ -66,9 +66,9 @@ var config = { ...@@ -66,9 +66,9 @@ var config = {
project_new: './projects/project_new.js', project_new: './projects/project_new.js',
prometheus_metrics: './prometheus_metrics', prometheus_metrics: './prometheus_metrics',
protected_branches: './protected_branches', protected_branches: './protected_branches',
ee_protected_branches: './protected_branches/ee', ee_protected_branches: 'ee/protected_branches',
protected_tags: './protected_tags', protected_tags: './protected_tags',
ee_protected_tags: './protected_tags/ee', ee_protected_tags: 'ee/protected_tags',
service_desk: './projects/settings_service_desk/service_desk_bundle.js', service_desk: './projects/settings_service_desk/service_desk_bundle.js',
sidebar: './sidebar/sidebar_bundle.js', sidebar: './sidebar/sidebar_bundle.js',
schedule_form: './pipeline_schedules/pipeline_schedule_form_bundle.js', schedule_form: './pipeline_schedules/pipeline_schedule_form_bundle.js',
...@@ -214,6 +214,7 @@ var config = { ...@@ -214,6 +214,7 @@ var config = {
resolve: { resolve: {
extensions: ['.js'], extensions: ['.js'],
alias: { alias: {
'ee': path.join(ROOT_PATH, 'ee/app/assets/javascripts'),
'~': path.join(ROOT_PATH, 'app/assets/javascripts'), '~': path.join(ROOT_PATH, 'app/assets/javascripts'),
'emojis': path.join(ROOT_PATH, 'fixtures/emojis'), 'emojis': path.join(ROOT_PATH, 'fixtures/emojis'),
'empty_states': path.join(ROOT_PATH, 'app/views/shared/empty_states'), 'empty_states': path.join(ROOT_PATH, 'app/views/shared/empty_states'),
......
...@@ -17,19 +17,19 @@ as much as possible. ...@@ -17,19 +17,19 @@ as much as possible.
Place EE-specific controllers, finders, helpers, mailers, models, policies, Place EE-specific controllers, finders, helpers, mailers, models, policies,
serializers/entities, services, validators and workers in the top-level serializers/entities, services, validators and workers in the top-level
`EE` module namespace, and in a specific `/ee/` sub-folder: `EE` module namespace, and in the `ee/` specific sub-directory:
- `app/controllers/ee/foos_controller.rb` - `ee/app/controllers/ee/foos_controller.rb`
- `app/finders/ee/foos_finder.rb` - `ee/app/finders/ee/foos_finder.rb`
- `app/helpers/ee/foos_helper.rb` - `ee/app/helpers/ee/foos_helper.rb`
- `app/mailers/ee/foos_mailer.rb` - `ee/app/mailers/ee/foos_mailer.rb`
- `app/models/ee/foo.rb` - `ee/app/models/ee/foo.rb`
- `app/policies/ee/foo_policy.rb` - `ee/app/policies/ee/foo_policy.rb`
- `app/serializers/ee/foo_entity.rb` - `ee/app/serializers/ee/foo_entity.rb`
- `app/serializers/ee/foo_serializer.rb` - `ee/app/serializers/ee/foo_serializer.rb`
- `app/services/ee/foo/create_service.rb` - `ee/app/services/ee/foo/create_service.rb`
- `app/validators/ee/foo_attr_validator.rb` - `ee/app/validators/ee/foo_attr_validator.rb`
- `app/workers/ee/foo_worker.rb` - `ee/app/workers/ee/foo_worker.rb`
If you modify an existing part of a CE controller, model, service, worker etc. If you modify an existing part of a CE controller, model, service, worker etc.
one simple solution is to use the `prepend` strategy ([presented below](#overriding-ce-methods)). one simple solution is to use the `prepend` strategy ([presented below](#overriding-ce-methods)).
...@@ -58,7 +58,7 @@ class ApplicationController < ActionController::Base ...@@ -58,7 +58,7 @@ class ApplicationController < ActionController::Base
end end
module EE module EE
class ApplicationController class ApplicationController
def after_sign_out_path_for(resource) def after_sign_out_path_for(resource)
raise NotImplementedError unless defined?(super) raise NotImplementedError unless defined?(super)
...@@ -117,7 +117,7 @@ end ...@@ -117,7 +117,7 @@ end
EE-specific models should `extend EE::Model`. EE-specific models should `extend EE::Model`.
For example, if EE has a specific `Tanuki` model, you would For example, if EE has a specific `Tanuki` model, you would
place it in `app/models/ee/tanuki.rb`. place it in `ee/app/models/ee/tanuki.rb`.
#### Code in `app/views/` #### Code in `app/views/`
...@@ -136,7 +136,7 @@ Place EE-specific logic in the top-level `EE` module namespace. Namespace the ...@@ -136,7 +136,7 @@ Place EE-specific logic in the top-level `EE` module namespace. Namespace the
class beneath the `EE` module just as you would normally. class beneath the `EE` module just as you would normally.
For example, if CE has LDAP classes in `lib/gitlab/ldap/` then you would place For example, if CE has LDAP classes in `lib/gitlab/ldap/` then you would place
EE-specific LDAP classes in `lib/ee/gitlab/ldap`. EE-specific LDAP classes in `ee/lib/ee/gitlab/ldap`.
### Classes vs. Module Mixins ### Classes vs. Module Mixins
......
...@@ -11,7 +11,7 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns. ...@@ -11,7 +11,7 @@ See [our current .eslintrc][eslintrc] for specific rules and patterns.
#### ESlint #### ESlint
1. **Never** disable eslint rules unless you have a good reason. 1. **Never** disable eslint rules unless you have a good reason.
You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */` You may see a lot of legacy files with `/* eslint-disable some-rule, some-other-rule */`
at the top, but legacy files are a special case. Any time you develop a new feature or at the top, but legacy files are a special case. Any time you develop a new feature or
refactor an existing one, you should abide by the eslint rules. refactor an existing one, you should abide by the eslint rules.
...@@ -100,26 +100,44 @@ followed by any global declarations, then a blank newline prior to any imports o ...@@ -100,26 +100,44 @@ followed by any global declarations, then a blank newline prior to any imports o
export default Foo; export default Foo;
``` ```
1. Relative paths: Unless you are writing a test, always reference other scripts using 1. Relative paths: when importing a module in the same directory, a child
relative paths instead of `~` directory, or an immediate parent directory prefer relative paths. When
* In **app/assets/javascripts**: importing a module which is two or more levels up, prefer either `~/` or `ee/`
.
```javascript In **app/assets/javascripts/my-feature/subdir**:
// bad
import Foo from '~/foo'
// good ``` javascript
import Foo from '../foo'; // bad
``` import Foo from '~/my-feature/foo';
* In **spec/javascripts**: import Bar from '~/my-feature/subdir/bar';
import Bin from '~/my-feature/subdir/lib/bin';
```javascript // good
// bad import Foo from '../foo';
import Foo from '../../app/assets/javascripts/foo' import Bar from './bar';
import Bin from './lib/bin';
```
// good In **spec/javascripts**:
import Foo from '~/foo';
``` ``` javascript
// bad
import Foo from '../../app/assets/javascripts/my-feature/foo';
// good
import Foo from '~/my-feature/foo';
```
When referencing an **EE component**:
``` javascript
// bad
import Foo from '../../../../../ee/app/assets/javascripts/my-feature/ee-foo';
// good
import Foo from 'ee/my-feature/foo';
```
1. Avoid using IIFE. Although we have a lot of examples of files which wrap their 1. Avoid using IIFE. Although we have a lot of examples of files which wrap their
contents in IIFEs (immediately-invoked function expressions), contents in IIFEs (immediately-invoked function expressions),
......
/* global Flash */ /* global Flash */
import MRWidgetAuthor from '../../../components/mr_widget_author'; import MRWidgetAuthor from '~/vue_merge_request_widget/components/mr_widget_author';
import eventHub from '../../../event_hub'; import eventHub from '~/vue_merge_request_widget/event_hub';
export default { export default {
name: 'approvals-body', name: 'approvals-body',
......
/* global Flash */ /* global Flash */
import LinkToMemberAvatar from '~/vue_shared/components/link_to_member_avatar'; import LinkToMemberAvatar from '~/vue_shared/components/link_to_member_avatar';
import eventHub from '../../../event_hub'; import eventHub from '~/vue_merge_request_widget/event_hub';
export default { export default {
name: 'approvals-footer', name: 'approvals-footer',
......
<script> <script>
import successIcon from 'icons/_icon_status_success.svg'; import successIcon from 'icons/_icon_status_success.svg';
import errorIcon from 'icons/_icon_status_failed.svg'; import errorIcon from 'icons/_icon_status_failed.svg';
import loadingIcon from '~/vue_shared/components/loading_icon.vue';
import '~/lib/utils/text_utility';
import issuesBlock from './mr_widget_code_quality_issues.vue'; import issuesBlock from './mr_widget_code_quality_issues.vue';
import loadingIcon from '../../../vue_shared/components/loading_icon.vue';
import '../../../lib/utils/text_utility';
export default { export default {
name: 'MRWidgetCodeQuality', name: 'MRWidgetCodeQuality',
......
import eventHub from '../../../event_hub'; import eventHub from '~/vue_merge_request_widget/event_hub';
import ReadyToMergeState from '../../../components/states/mr_widget_ready_to_merge'; import ReadyToMergeState from '~/vue_merge_request_widget/components/states/mr_widget_ready_to_merge';
import SquashBeforeMerge from './mr_widget_squash_before_merge'; import SquashBeforeMerge from './mr_widget_squash_before_merge';
export default { export default {
......
/* global Flash */ /* global Flash */
import simplePoll from '~/lib/utils/simple_poll'; import simplePoll from '~/lib/utils/simple_poll';
import eventHub from '../../../event_hub'; import eventHub from '~/vue_merge_request_widget/event_hub';
export default { export default {
props: { props: {
......
import eventHub from '../../../event_hub'; import eventHub from '~/vue_merge_request_widget/event_hub';
import CESquashBeforeMerge from '../../components/states/mr_widget_squash_before_merge'; import CESquashBeforeMerge from '~/vue_merge_request_widget/components/states/mr_widget_squash_before_merge';
export default { export default {
extends: CESquashBeforeMerge, extends: CESquashBeforeMerge,
......
import CEWidgetOptions from '../mr_widget_options'; import CEWidgetOptions from '~/vue_merge_request_widget/mr_widget_options';
import WidgetApprovals from './components/approvals/mr_widget_approvals'; import WidgetApprovals from './components/approvals/mr_widget_approvals';
import GeoSecondaryNode from './components/states/mr_widget_secondary_geo_node'; import GeoSecondaryNode from './components/states/mr_widget_secondary_geo_node';
import RebaseState from './components/states/mr_widget_rebase'; import RebaseState from './components/states/mr_widget_rebase';
......
import Vue from 'vue'; import Vue from 'vue';
import CEWidgetService from '../../services/mr_widget_service'; import CEWidgetService from '~/vue_merge_request_widget/services/mr_widget_service';
export default class MRWidgetService extends CEWidgetService { export default class MRWidgetService extends CEWidgetService {
constructor(mr) { constructor(mr) {
......
import CEGetStateKey from '../../stores/get_state_key'; import CEGetStateKey from '~/vue_merge_request_widget/stores/get_state_key';
export default function (data) { export default function (data) {
if (this.isGeoSecondaryNode) { if (this.isGeoSecondaryNode) {
......
import CEMergeRequestStore from '../../stores/mr_widget_store'; import CEMergeRequestStore from '~/vue_merge_request_widget/stores/mr_widget_store';
export default class MergeRequestStore extends CEMergeRequestStore { export default class MergeRequestStore extends CEMergeRequestStore {
constructor(data) { constructor(data) {
......
import stateMaps from '../../stores/state_maps'; import stateMaps from '~/vue_merge_request_widget/stores/state_maps';
stateMaps.stateToComponentMap.geoSecondaryNode = 'mr-widget-geo-secondary-node'; stateMaps.stateToComponentMap.geoSecondaryNode = 'mr-widget-geo-secondary-node';
stateMaps.stateToComponentMap.rebase = 'mr-widget-rebase'; stateMaps.stateToComponentMap.rebase = 'mr-widget-rebase';
......
...@@ -2,6 +2,8 @@ class Admin::GeoNodesController < Admin::ApplicationController ...@@ -2,6 +2,8 @@ class Admin::GeoNodesController < Admin::ApplicationController
before_action :check_license, except: [:index, :destroy] before_action :check_license, except: [:index, :destroy]
before_action :load_node, only: [:edit, :update, :destroy, :repair, :toggle, :status] before_action :load_node, only: [:edit, :update, :destroy, :repair, :toggle, :status]
helper EE::GeoHelper
def index def index
@nodes = GeoNode.all.order(:id) @nodes = GeoNode.all.order(:id)
@node = GeoNode.new @node = GeoNode.new
......
# Define the EE module
module EE
end
{ {
"allOf": [ "allOf": [
{ "$ref": "../issues.json" }, { "$ref": "../../../../../../fixtures/api/schemas/public_api/v4/issues.json" },
{ {
"properties": { "properties": {
"weight": { "type": ["integer", "null"] } "weight": { "type": ["integer", "null"] }
......
require 'spec_helper' require 'spec_helper'
require_relative '../../email_shared_blocks' require Rails.root.join('spec/lib/gitlab/email/email_shared_blocks')
describe Gitlab::Email::Handler::EE::ServiceDeskHandler do describe Gitlab::Email::Handler::EE::ServiceDeskHandler do
include_context :email_shared_context include_context :email_shared_context
......
require 'spec_helper' require 'spec_helper'
describe Gitlab::VisibilityLevel do # rubocop:disable RSpec/FilePath describe Gitlab::VisibilityLevel do
describe '.levels_for_user' do describe '.levels_for_user' do
it 'returns all levels for an auditor' do it 'returns all levels for an auditor' do
user = build(:user, :auditor) user = build(:user, :auditor)
......
...@@ -34,7 +34,7 @@ describe API::Issues, :mailer do # rubocop:disable RSpec/FilePath ...@@ -34,7 +34,7 @@ describe API::Issues, :mailer do # rubocop:disable RSpec/FilePath
get api('/issues', user) get api('/issues', user)
expect(response).to have_http_status(200) expect(response).to have_http_status(200)
expect(response).to match_response_schema('public_api/v4/ee/issues') expect(response).to match_response_schema('public_api/v4/issues', dir: 'ee')
end end
end end
end end
......
import Vue from 'vue'; import Vue from 'vue';
import ApprovalsBody from '~/vue_merge_request_widget/ee/components/approvals/approvals_body'; import ApprovalsBody from 'ee/vue_merge_request_widget/components/approvals/approvals_body';
(() => { (() => {
gl.ApprovalsStore = { gl.ApprovalsStore = {
......
import Vue from 'vue'; import Vue from 'vue';
import pendingAvatarSvg from 'icons/_icon_dotted_circle.svg'; import pendingAvatarSvg from 'icons/_icon_dotted_circle.svg';
import ApprovalsFooter from '~/vue_merge_request_widget/ee/components/approvals/approvals_footer'; import ApprovalsFooter from 'ee/vue_merge_request_widget/components/approvals/approvals_footer';
(() => { (() => {
gl.ApprovalsStore = { gl.ApprovalsStore = {
......
import Vue from 'vue'; import Vue from 'vue';
import mrWidgetCodeQualityIssues from '~/vue_merge_request_widget/ee/components/mr_widget_code_quality_issues.vue'; import mrWidgetCodeQualityIssues from 'ee/vue_merge_request_widget/components/mr_widget_code_quality_issues.vue';
describe('Merge Request Code Quality Issues', () => { describe('Merge Request Code Quality Issues', () => {
let vm; let vm;
......
import Vue from 'vue'; import Vue from 'vue';
import mrWidgetCodeQuality from '~/vue_merge_request_widget/ee/components/mr_widget_code_quality.vue'; import mrWidgetCodeQuality from 'ee/vue_merge_request_widget/components/mr_widget_code_quality.vue';
import Store from '~/vue_merge_request_widget/ee/stores/mr_widget_store'; import Store from 'ee/vue_merge_request_widget/stores/mr_widget_store';
import Service from '~/vue_merge_request_widget/ee/services/mr_widget_service'; import Service from 'ee/vue_merge_request_widget/services/mr_widget_service';
import mockData, { baseIssues, headIssues } from '../mock_data'; import mockData, { baseIssues, headIssues } from '../mock_data';
describe('Merge Request Code Quality', () => { describe('Merge Request Code Quality', () => {
......
import MergeRequestStore from '~/vue_merge_request_widget/ee/stores/mr_widget_store'; import MergeRequestStore from 'ee/vue_merge_request_widget/stores/mr_widget_store';
import mockData, { headIssues, baseIssues } from '../mock_data'; import mockData, { headIssues, baseIssues } from '../mock_data';
describe('MergeRequestStore', () => { describe('MergeRequestStore', () => {
......
...@@ -29,6 +29,9 @@ end ...@@ -29,6 +29,9 @@ end
# require rainbow gem String monkeypatch, so we can test SystemChecks # require rainbow gem String monkeypatch, so we can test SystemChecks
require 'rainbow/ext/string' require 'rainbow/ext/string'
# EE specific support
Dir[Rails.root.join("spec/ee/support/**/*.rb")].each { |f| require f }
# Requires supporting ruby files with custom matchers and macros, etc, # Requires supporting ruby files with custom matchers and macros, etc,
# in spec/support/ and its subdirectories. # in spec/support/ and its subdirectories.
Dir[Rails.root.join("spec/support/**/*.rb")].each { |f| require f } Dir[Rails.root.join("spec/support/**/*.rb")].each { |f| require f }
......
...@@ -229,7 +229,7 @@ shared_examples_for 'group and project milestones' do |route_definition| ...@@ -229,7 +229,7 @@ shared_examples_for 'group and project milestones' do |route_definition|
get api(issues_route, user) get api(issues_route, user)
expect(response).to have_http_status(200) expect(response).to have_http_status(200)
expect(response).to match_response_schema('public_api/v4/ee/issues') expect(response).to match_response_schema('public_api/v4/issues', dir: 'ee')
end end
it 'returns a 401 error if user not authenticated' do it 'returns a 401 error if user not authenticated' do
......
def schema_path(schema) module SchemaPath
schema_directory = "#{Dir.pwd}/spec/fixtures/api/schemas" def self.expand(schema, dir = '')
"#{schema_directory}/#{schema}.json" Rails.root.join('spec', dir, "fixtures/api/schemas/#{schema}.json").to_s
end
end end
RSpec::Matchers.define :match_response_schema do |schema, **options| RSpec::Matchers.define :match_response_schema do |schema, dir: '', **options|
match do |response| match do |response|
@errors = JSON::Validator.fully_validate(schema_path(schema), response.body, options) @errors = JSON::Validator.fully_validate(
SchemaPath.expand(schema, dir), response.body, options)
@errors.empty? @errors.empty?
end end
failure_message do |response| failure_message do |response|
"didn't match the schema defined by #{schema_path(schema)}" \ "didn't match the schema defined by #{SchemaPath.expand(schema, dir)}" \
" The validation errors were:\n#{@errors.join("\n")}" " The validation errors were:\n#{@errors.join("\n")}"
end end
end end
RSpec::Matchers.define :match_schema do |schema, **options| RSpec::Matchers.define :match_schema do |schema, dir: '', **options|
match do |data| match do |data|
JSON::Validator.validate!(schema_path(schema), data, options) JSON::Validator.validate!(SchemaPath.expand(schema, dir), data, options)
end end
end end
require_relative 'ee/ldap_helpers'
module LdapHelpers module LdapHelpers
include EE::LdapHelpers include EE::LdapHelpers
......
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