diff --git a/app/controllers/projects/web_ide_schemas_controller.rb b/app/controllers/projects/web_ide_schemas_controller.rb
new file mode 100644
index 0000000000000000000000000000000000000000..3d16a6fafd42a681c7593bdf94a23b683c4643af
--- /dev/null
+++ b/app/controllers/projects/web_ide_schemas_controller.rb
@@ -0,0 +1,25 @@
+# frozen_string_literal: true
+
+class Projects::WebIdeSchemasController < Projects::ApplicationController
+  before_action :authenticate_user!
+
+  def show
+    return respond_422 unless branch_sha
+
+    result = ::Ide::SchemasConfigService.new(project, current_user, sha: branch_sha, filename: params[:filename]).execute
+
+    if result[:status] == :success
+      render json: result[:schema]
+    else
+      render json: result, status: :unprocessable_entity
+    end
+  end
+
+  private
+
+  def branch_sha
+    return unless params[:branch].present?
+
+    project.commit(params[:branch])&.id
+  end
+end
diff --git a/app/controllers/projects/web_ide_terminals_controller.rb b/app/controllers/projects/web_ide_terminals_controller.rb
index 08ea5c4bca841dfe0a9520a759f6bb493980ed53..76bcaa9e80c3c23503247dd6edba9bceda346590 100644
--- a/app/controllers/projects/web_ide_terminals_controller.rb
+++ b/app/controllers/projects/web_ide_terminals_controller.rb
@@ -11,7 +11,7 @@ class Projects::WebIdeTerminalsController < Projects::ApplicationController
   def check_config
     return respond_422 unless branch_sha
 
-    result = ::Ci::WebIdeConfigService.new(project, current_user, sha: branch_sha).execute
+    result = ::Ide::TerminalConfigService.new(project, current_user, sha: branch_sha).execute
 
     if result[:status] == :success
       head :ok
diff --git a/app/services/ci/create_web_ide_terminal_service.rb b/app/services/ci/create_web_ide_terminal_service.rb
index 4f1bf0447d2b7227ecf3289481af63fd3d8c237c..a78281aed161d067f3635968ed52198d80f7071f 100644
--- a/app/services/ci/create_web_ide_terminal_service.rb
+++ b/app/services/ci/create_web_ide_terminal_service.rb
@@ -70,7 +70,7 @@ def terminal_build_seed
     end
 
     def load_terminal_config!
-      result = ::Ci::WebIdeConfigService.new(project, current_user, sha: sha).execute
+      result = ::Ide::TerminalConfigService.new(project, current_user, sha: sha).execute
       raise TerminalCreationError, result[:message] if result[:status] != :success
 
       @terminal = result[:terminal]
diff --git a/app/services/ci/web_ide_config_service.rb b/app/services/ide/base_config_service.rb
similarity index 88%
rename from app/services/ci/web_ide_config_service.rb
rename to app/services/ide/base_config_service.rb
index ade9132f419af6ec8a5ad2f7752435ea151ff175..1f8d5c17584b56606a96561acf020dafc95972be 100644
--- a/app/services/ci/web_ide_config_service.rb
+++ b/app/services/ide/base_config_service.rb
@@ -1,9 +1,7 @@
 # frozen_string_literal: true
 
-module Ci
-  class WebIdeConfigService < ::BaseService
-    include ::Gitlab::Utils::StrongMemoize
-
+module Ide
+  class BaseConfigService < ::BaseService
     ValidationError = Class.new(StandardError)
 
     WEBIDE_CONFIG_FILE = '.gitlab/.gitlab-webide.yml'.freeze
@@ -11,15 +9,21 @@ class WebIdeConfigService < ::BaseService
     attr_reader :config, :config_content
 
     def execute
-      check_access!
-      load_config_content!
-      load_config!
+      check_access_and_load_config!
 
-      success(terminal: config.terminal_value)
+      success
     rescue ValidationError => e
       error(e.message)
     end
 
+    protected
+
+    def check_access_and_load_config!
+      check_access!
+      load_config_content!
+      load_config!
+    end
+
     private
 
     def check_access!
diff --git a/app/services/ide/schemas_config_service.rb b/app/services/ide/schemas_config_service.rb
new file mode 100644
index 0000000000000000000000000000000000000000..8d2ce97103d216f0efc4cae93e6f495e60a33888
--- /dev/null
+++ b/app/services/ide/schemas_config_service.rb
@@ -0,0 +1,49 @@
+# frozen_string_literal: true
+
+module Ide
+  class SchemasConfigService < ::Ide::BaseConfigService
+    PREDEFINED_SCHEMAS = [{
+      uri: 'https://json.schemastore.org/gitlab-ci',
+      match: ['*.gitlab-ci.yml']
+    }].freeze
+
+    def execute
+      schema = predefined_schema_for(params[:filename]) || {}
+      success(schema: schema)
+    rescue => e
+      error(e.message)
+    end
+
+    private
+
+    def find_schema(filename, schemas)
+      match_flags = ::File::FNM_DOTMATCH | ::File::FNM_PATHNAME
+
+      schemas.each do |schema|
+        match = schema[:match].any? { |pattern| ::File.fnmatch?(pattern, filename, match_flags) }
+
+        return Gitlab::Json.parse(get_cached(schema[:uri])) if match
+      end
+
+      nil
+    end
+
+    def predefined_schema_for(filename)
+      find_schema(filename, predefined_schemas)
+    end
+
+    def predefined_schemas
+      return PREDEFINED_SCHEMAS if Feature.enabled?(:schema_linting)
+
+      []
+    end
+
+    def get_cached(url)
+      Rails.cache.fetch("services:ide:schema:#{url}", expires_in: 1.day) do
+        Gitlab::HTTP.get(url).body
+      end
+    end
+  end
+end
+
+Ide::SchemasConfigService.prepend_if_ee('::EE::Ide::SchemasConfigService')
diff --git a/app/services/ide/terminal_config_service.rb b/app/services/ide/terminal_config_service.rb
new file mode 100644
index 0000000000000000000000000000000000000000..318df3436c43b6ce1b2727bbd7f844838b2a3e85
--- /dev/null
+++ b/app/services/ide/terminal_config_service.rb
@@ -0,0 +1,13 @@
+# frozen_string_literal: true
+
+module Ide
+  class TerminalConfigService < ::Ide::BaseConfigService
+    private
+
+    def success(pass_back = {})
+      result = super(pass_back)
+      result[:terminal] = config.terminal_value
+      result
+    end
+  end
+end
diff --git a/config/routes/project.rb b/config/routes/project.rb
index 3361193581e8ee966985187972154f61752f6810..24b44646d9504cf2c04e7ddc3595f69f236cddf1 100644
--- a/config/routes/project.rb
+++ b/config/routes/project.rb
@@ -378,6 +378,11 @@
           post :reset_token
         end
         resources :feature_flags_user_lists, param: :iid, only: [:new, :edit, :show]
+
+        get '/schema/:branch/*filename',
+          to: 'web_ide_schemas#show',
+          format: false,
+          as: :schema
       end
       # End of the /-/ scope.
 
diff --git a/ee/app/models/license.rb b/ee/app/models/license.rb
index 5c238367ea6b55c8bdb01b1274a78ddd61ba9ee6..4209f523bd72f0a79defca36edb46e6fb8a0a3ba 100644
--- a/ee/app/models/license.rb
+++ b/ee/app/models/license.rb
@@ -85,6 +85,7 @@ class License < ApplicationRecord
     group_project_templates
     group_repository_analytics
     group_saml
+    ide_schema_config
     issues_analytics
     jira_issues_integration
     ldap_group_sync_filter
diff --git a/ee/app/services/ee/ide/schemas_config_service.rb b/ee/app/services/ee/ide/schemas_config_service.rb
new file mode 100644
index 0000000000000000000000000000000000000000..44189b9e988ba301f720777bab333c070922bcf3
--- /dev/null
+++ b/ee/app/services/ee/ide/schemas_config_service.rb
@@ -0,0 +1,28 @@
+# frozen_string_literal: true
+
+module EE
+  module Ide
+    module SchemasConfigService
+      extend ::Gitlab::Utils::Override
+
+      override :execute
+      def execute
+        result = super
+        return result if result[:status] == :success && !result[:schema].empty?
+
+        check_access_and_load_config!
+        success(schema: schema_from_config_for(params[:filename]) || {})
+      rescue => e
+        error(e.message)
+      end
+
+      private
+
+      def schema_from_config_for(filename)
+        return {} unless project.feature_available?(:ide_schema_config)
+
+        find_schema(filename, config.schemas_value || [])
+      end
+    end
+  end
+end
diff --git a/ee/lib/ee/gitlab/web_ide/config/entry/global.rb b/ee/lib/ee/gitlab/web_ide/config/entry/global.rb
new file mode 100644
index 0000000000000000000000000000000000000000..4fdcd6bec06ca30dc584c390ffe1ab81ef9475be
--- /dev/null
+++ b/ee/lib/ee/gitlab/web_ide/config/entry/global.rb
@@ -0,0 +1,26 @@
+# frozen_string_literal: true
+
+module EE
+  module Gitlab
+    module WebIde
+      module Config
+        module Entry
+          module Global
+            extend ActiveSupport::Concern
+
+            class_methods do
+              def allowed_keys
+                %i[terminal schemas].freeze
+              end
+            end
+
+            prepended do
+              entry :schemas, ::Gitlab::WebIde::Config::Entry::Schemas,
+                description: 'Configuration of JSON/YAML schemas.'
+            end
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/ee/lib/gitlab/web_ide/config/entry/schema.rb b/ee/lib/gitlab/web_ide/config/entry/schema.rb
new file mode 100644
index 0000000000000000000000000000000000000000..5015d04cba4f9916adaf586956b67b1e78c38e1a
--- /dev/null
+++ b/ee/lib/gitlab/web_ide/config/entry/schema.rb
@@ -0,0 +1,40 @@
+# frozen_string_literal: true
+
+module Gitlab
+  module WebIde
+    class Config
+      module Entry
+        ##
+        # Entry that represents a JSON/YAML schema.
+        #
+        class Schema < ::Gitlab::Config::Entry::Node
+          include ::Gitlab::Config::Entry::Configurable
+          include ::Gitlab::Config::Entry::Attributable
+
+          ALLOWED_KEYS = %i[uri match].freeze
+
+          validations do
+            validates :config, allowed_keys: ALLOWED_KEYS
+          end
+
+          entry :uri, Entry::Schema::Uri,
+            description: 'The URI of the schema.'
+
+          entry :match, Entry::Schema::Match,
+            description: 'A list of glob expressions to match against the target file.'
+
+          def value
+            to_hash.compact
+          end
+
+          private
+
+          def to_hash
+            { uri: uri_value,
+              match: match_value || [] }
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/ee/lib/gitlab/web_ide/config/entry/schema/match.rb b/ee/lib/gitlab/web_ide/config/entry/schema/match.rb
new file mode 100644
index 0000000000000000000000000000000000000000..90d1818513f50e6da6fb23b11c1905b7877fd3bf
--- /dev/null
+++ b/ee/lib/gitlab/web_ide/config/entry/schema/match.rb
@@ -0,0 +1,26 @@
+# frozen_string_literal: true
+
+module Gitlab
+  module WebIde
+    class Config
+      module Entry
+        class Schema
+          ##
+          # Entry that represents a list of glob expressions to match against the target file.
+          #
+          class Match < ::Gitlab::Config::Entry::Node
+            include ::Gitlab::Config::Entry::Validatable
+
+            validations do
+              validates :config, array_of_strings: true, presence: true
+            end
+
+            def self.default
+              []
+            end
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/ee/lib/gitlab/web_ide/config/entry/schema/uri.rb b/ee/lib/gitlab/web_ide/config/entry/schema/uri.rb
new file mode 100644
index 0000000000000000000000000000000000000000..901d7c5e07ade1403f69c3731f92a3548a851b8c
--- /dev/null
+++ b/ee/lib/gitlab/web_ide/config/entry/schema/uri.rb
@@ -0,0 +1,26 @@
+# frozen_string_literal: true
+
+module Gitlab
+  module WebIde
+    class Config
+      module Entry
+        class Schema
+          ##
+          # Entry that represents the URI of a schema
+          #
+          class Uri < ::Gitlab::Config::Entry::Node
+            include ::Gitlab::Config::Entry::Validatable
+
+            validations do
+              validates :config, presence: true, type: String
+            end
+
+            def self.default
+              ''
+            end
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/ee/lib/gitlab/web_ide/config/entry/schemas.rb b/ee/lib/gitlab/web_ide/config/entry/schemas.rb
new file mode 100644
index 0000000000000000000000000000000000000000..a693fa273587b272d90177e122ec28ff75c3c090
--- /dev/null
+++ b/ee/lib/gitlab/web_ide/config/entry/schemas.rb
@@ -0,0 +1,27 @@
+# frozen_string_literal: true
+
+module Gitlab
+  module WebIde
+    class Config
+      module Entry
+        ##
+        # Entry that represents an array of JSON/YAML schemas
+        #
+        class Schemas < ::Gitlab::Config::Entry::Node
+          include ::Gitlab::Config::Entry::Configurable
+          include ::Gitlab::Config::Entry::Validatable
+
+          entry :schema, Entry::Schema, description: 'A JSON/YAML schema definition'
+
+          validations do
+            validates :config, type: Array
+          end
+
+          def skip_config_hash_validation?
+            true
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/ee/spec/lib/ee/gitlab/web_ide/config/entry/global_spec.rb b/ee/spec/lib/ee/gitlab/web_ide/config/entry/global_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..657f946ed784e4a5309e70ec1fbe373260827e7b
--- /dev/null
+++ b/ee/spec/lib/ee/gitlab/web_ide/config/entry/global_spec.rb
@@ -0,0 +1,80 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Gitlab::WebIde::Config::Entry::Global do
+  let(:global) { described_class.new(hash) }
+
+  describe '.nodes' do
+    context 'when filtering all the entry/node names' do
+      it 'contains the expected node names' do
+        expect(described_class.nodes.keys).to match_array(%i[terminal schemas])
+      end
+    end
+  end
+
+  context 'when configuration is valid' do
+    context 'when some entries defined' do
+      let(:hash) do
+        {
+          terminal: { before_script: ['ls'], variables: {}, script: 'sleep 10s', services: ['mysql'] },
+          schemas: [{ uri: 'https://someurl.com', match: ['*.gitlab-ci.yml'] }]
+        }
+      end
+
+      describe '#compose!' do
+        before do
+          global.compose!
+        end
+
+        it 'creates nodes hash' do
+          expect(global.descendants).to be_an Array
+        end
+
+        it 'creates node object for each entry' do
+          expect(global.descendants.count).to eq 2
+        end
+
+        it 'creates node object using valid class' do
+          expect(global.descendants.first)
+            .to be_an_instance_of Gitlab::WebIde::Config::Entry::Terminal
+          expect(global.descendants.second)
+            .to be_an_instance_of Gitlab::WebIde::Config::Entry::Schemas
+        end
+
+        it 'sets correct description for nodes' do
+          expect(global.descendants.first.description)
+            .to eq 'Configuration of the webide terminal.'
+          expect(global.descendants.second.description)
+            .to eq 'Configuration of JSON/YAML schemas.'
+        end
+      end
+
+      context 'when not composed' do
+        describe '#schemas_value' do
+          it 'returns nil' do
+            expect(global.schemas_value).to be nil
+          end
+        end
+      end
+
+      context 'when composed' do
+        before do
+          global.compose!
+        end
+
+        describe '#errors' do
+          it 'has no errors' do
+            expect(global.errors).to be_empty
+          end
+        end
+
+        describe '#schemas_value' do
+          it 'returns correct value for schemas' do
+            expect(global.schemas_value).to eq([{ uri: 'https://someurl.com', match: ['*.gitlab-ci.yml'] }])
+          end
+        end
+      end
+    end
+  end
+end
diff --git a/ee/spec/lib/gitlab/web_ide/config/entry/schema/match_spec.rb b/ee/spec/lib/gitlab/web_ide/config/entry/schema/match_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..c215f30f54b8e79f8935cef126238167446ad0c9
--- /dev/null
+++ b/ee/spec/lib/gitlab/web_ide/config/entry/schema/match_spec.rb
@@ -0,0 +1,40 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Gitlab::WebIde::Config::Entry::Schema::Match do
+  let(:match) { described_class.new(config) }
+
+  describe 'validations' do
+    context 'when match config value is correct' do
+      let(:config) { ['*.json'] }
+
+      describe '#value' do
+        it 'returns the match glob pattern defined' do
+          expect(match.value).to eq config
+        end
+      end
+
+      describe '#valid?' do
+        it 'is valid' do
+          expect(match).to be_valid
+        end
+      end
+    end
+
+    context 'when value has a wrong type' do
+      let(:config) { { test: true } }
+
+      it 'reports errors about wrong type' do
+        expect(match.errors)
+          .to include 'match config should be an array of strings'
+      end
+    end
+  end
+
+  describe '.default' do
+    it 'returns empty array' do
+      expect(described_class.default).to eq []
+    end
+  end
+end
diff --git a/ee/spec/lib/gitlab/web_ide/config/entry/schema/uri_spec.rb b/ee/spec/lib/gitlab/web_ide/config/entry/schema/uri_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..35ef1eaa8cb082b3357a1369e7a631562b451514
--- /dev/null
+++ b/ee/spec/lib/gitlab/web_ide/config/entry/schema/uri_spec.rb
@@ -0,0 +1,40 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Gitlab::WebIde::Config::Entry::Schema::Uri do
+  let(:uri) { described_class.new(config) }
+
+  describe 'validations' do
+    context 'when uri config value is correct' do
+      let(:config) { 'https://someurl.com' }
+
+      describe '#value' do
+        it 'returns the url defined' do
+          expect(uri.value).to eq config
+        end
+      end
+
+      describe '#valid?' do
+        it 'is valid' do
+          expect(uri).to be_valid
+        end
+      end
+    end
+
+    context 'when value has a wrong type' do
+      let(:config) { { test: true } }
+
+      it 'reports errors about wrong type' do
+        expect(uri.errors)
+          .to include 'uri config should be a string'
+      end
+    end
+  end
+
+  describe '.default' do
+    it 'returns empty string' do
+      expect(described_class.default).to eq ''
+    end
+  end
+end
diff --git a/ee/spec/lib/gitlab/web_ide/config/entry/schema_spec.rb b/ee/spec/lib/gitlab/web_ide/config/entry/schema_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..f189f1a3a12b9a041303b09623e0c27c1d752d8d
--- /dev/null
+++ b/ee/spec/lib/gitlab/web_ide/config/entry/schema_spec.rb
@@ -0,0 +1,144 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Gitlab::WebIde::Config::Entry::Schema do
+  let(:schema) { described_class.new(hash) }
+
+  describe '.nodes' do
+    it 'returns a hash' do
+      expect(described_class.nodes).to be_a(Hash)
+    end
+
+    context 'when filtering all the entry/node names' do
+      it 'contains the expected node names' do
+        expect(described_class.nodes.keys)
+          .to match_array(%i[uri match])
+      end
+    end
+  end
+
+  context 'when configuration is valid' do
+    context 'when some entries defined' do
+      let(:hash) do
+        { uri: 'https://someurl.com', match: ['*.gitlab-ci.yml'] }
+      end
+
+      describe '#compose!' do
+        before do
+          schema.compose!
+        end
+
+        it 'creates node object for each entry' do
+          expect(schema.descendants.count).to eq 2
+        end
+
+        it 'creates node object using valid class' do
+          expect(schema.descendants.first)
+            .to be_an_instance_of Gitlab::WebIde::Config::Entry::Schema::Uri
+
+          expect(schema.descendants.second)
+            .to be_an_instance_of Gitlab::WebIde::Config::Entry::Schema::Match
+        end
+
+        it 'sets correct description for nodes' do
+          expect(schema.descendants.first.description)
+            .to eq 'The URI of the schema.'
+
+          expect(schema.descendants.second.description)
+            .to eq 'A list of glob expressions to match against the target file.'
+        end
+
+        describe '#leaf?' do
+          it 'is not leaf' do
+            expect(schema).not_to be_leaf
+          end
+        end
+      end
+
+      context 'when composed' do
+        before do
+          schema.compose!
+        end
+
+        describe '#errors' do
+          it 'has no errors' do
+            expect(schema.errors).to be_empty
+          end
+        end
+
+        describe '#uri_value' do
+          it 'returns correct uri' do
+            expect(schema.uri_value).to eq('https://someurl.com')
+          end
+        end
+
+        describe '#match_value' do
+          it 'returns correct value for schemas' do
+            expect(schema.match_value).to eq(['*.gitlab-ci.yml'])
+          end
+        end
+      end
+    end
+  end
+
+  context 'when configuration is not valid' do
+    before do
+      schema.compose!
+    end
+
+    context 'when the config does not have all the required entries' do
+      let(:hash) do
+        {}
+      end
+
+      describe '#errors' do
+        it 'reports errors about the invalid entries' do
+          expect(schema.errors)
+            .to eq [
+              "uri config can't be blank",
+              "match config can't be blank"
+            ]
+        end
+      end
+    end
+
+    context 'when the config has invalid entries' do
+      let(:hash) do
+        { uri: 1, match: [2] }
+      end
+
+      describe '#errors' do
+        it 'reports errors about the invalid entries' do
+          expect(schema.errors)
+            .to eq [
+              "uri config should be a string",
+              "match config should be an array of strings"
+            ]
+        end
+      end
+    end
+  end
+
+  context 'when value is not a hash' do
+    let(:hash) { [] }
+
+    describe '#valid?' do
+      it 'is not valid' do
+        expect(schema).not_to be_valid
+      end
+    end
+
+    describe '#errors' do
+      it 'returns error about invalid type' do
+        expect(schema.errors.first).to match /should be a hash/
+      end
+    end
+  end
+
+  describe '#specified?' do
+    it 'is concrete entry that is defined' do
+      expect(schema.specified?).to be true
+    end
+  end
+end
diff --git a/ee/spec/lib/gitlab/web_ide/config/entry/schemas_spec.rb b/ee/spec/lib/gitlab/web_ide/config/entry/schemas_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..69729704964ca2a29ee9833b284d7412769a675e
--- /dev/null
+++ b/ee/spec/lib/gitlab/web_ide/config/entry/schemas_spec.rb
@@ -0,0 +1,66 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Gitlab::WebIde::Config::Entry::Schemas do
+  let(:entry) { described_class.new(config) }
+
+  describe 'validations' do
+    before do
+      entry.compose!
+    end
+
+    context 'when entry config value is correct' do
+      let(:config) { [{ uri: 'http://test.uri', match: '*-config.yml' }] }
+
+      it 'is valid' do
+        expect(entry).to be_valid
+      end
+    end
+
+    context 'when entry config value is incorrect' do
+      let(:config) { { incorrect: 'schemas config' } }
+
+      it 'is not valid' do
+        expect(entry).not_to be_valid
+
+        expect(entry.errors.first)
+          .to match /schema/
+      end
+
+      describe '#errors' do
+        it 'reports error about a config type' do
+          expect(entry.errors)
+            .to include 'schemas config should be a array'
+        end
+      end
+    end
+  end
+
+  context 'when composed' do
+    before do
+      entry.compose!
+    end
+
+    describe '#value' do
+      context 'when entry is correct' do
+        let(:config) do
+          [
+            {
+              uri: 'http://test.uri',
+              match: '*-config.yml'
+            }
+          ]
+        end
+
+        it 'returns correct value' do
+          expect(entry.value)
+            .to eq([{
+              uri: 'http://test.uri',
+              match: '*-config.yml'
+            }])
+        end
+      end
+    end
+  end
+end
diff --git a/ee/spec/services/ide/schemas_config_service_spec.rb b/ee/spec/services/ide/schemas_config_service_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..745fc683a066d7bbcd448604d9577b4f4a2c205f
--- /dev/null
+++ b/ee/spec/services/ide/schemas_config_service_spec.rb
@@ -0,0 +1,130 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Ide::SchemasConfigService do
+  let_it_be(:project) { create(:project, :repository) }
+  let_it_be(:user) { create(:user) }
+  let(:sha) { 'sha' }
+  let(:filename) { 'sample.yml' }
+  let(:schema_content) { double(body: '{"title":"Sample schema"}') }
+
+  describe '#execute' do
+    before do
+      project.add_developer(user)
+
+      allow(project.repository).to receive(:blob_data_at).with('sha', anything) do
+        config_content
+      end
+
+      allow(Gitlab::HTTP).to receive(:get).with(anything) do
+        schema_content
+      end
+    end
+
+    subject { described_class.new(project, user, sha: sha, filename: filename).execute }
+
+    context 'content is not valid' do
+      let(:config_content) { 'invalid content' }
+
+      it 'returns an error' do
+        is_expected.to include(
+          status: :error,
+          message: "Invalid configuration format")
+      end
+    end
+
+    context 'when a predefined schema exists for the given filename' do
+      let(:filename) { '.gitlab-ci.yml' }
+
+      before do
+        stub_feature_flags(schema_linting: true)
+      end
+
+      context 'with valid config content' do
+        let(:config_content) { 'schemas: [{uri: "https://someurl.com", match: ["*.yml"]}]' }
+
+        it 'uses predefined schema matches' do
+          expect(Gitlab::HTTP).to receive(:get).with('https://json.schemastore.org/gitlab-ci')
+          expect(Gitlab::HTTP).not_to receive(:get).with('https://someurl.com')
+
+          expect(subject[:schema]['title']).to eq "Sample schema"
+        end
+      end
+
+      context 'with invalid config content' do
+        let(:config_content) { '' }
+
+        it 'uses predefined schema matches' do
+          expect(Gitlab::HTTP).to receive(:get).with('https://json.schemastore.org/gitlab-ci')
+
+          expect(subject[:schema]['title']).to eq "Sample schema"
+        end
+      end
+    end
+
+    context 'no schemas are defined' do
+      let(:config_content) { '{}' }
+
+      it 'returns success with an empty object' do
+        is_expected.to include(
+          status: :success,
+          schema: {})
+      end
+    end
+
+    context 'feature :ide_schema_config is not available' do
+      let(:config_content) { 'schemas: [{uri: "https://someurl.com", match: ["*.yml"]}]' }
+
+      it 'returns empty object, despite config being defined' do
+        expect(Gitlab::HTTP).not_to receive(:get).with("https://someurl.com")
+        expect(subject[:schema]).to eq({})
+      end
+    end
+
+    context 'feature :ide_schema_config is available' do
+      before do
+        allow(project).to receive(:feature_available?).with(:ide_schema_config) { true }
+      end
+
+      context 'schemas are defined and a matching schema is found and valid' do
+        let(:config_content) { 'schemas: [{uri: "https://someurl.com", match: ["*.yml"]}]' }
+
+        it 'returns schema successfully' do
+          expect(Gitlab::HTTP).to receive(:get).with("https://someurl.com")
+          expect(subject[:schema]['title']).to eq "Sample schema"
+        end
+      end
+
+      context 'schemas are defined and a matching schema is found and but the schema is not a valid JSON' do
+        let(:config_content) { 'schemas: [{uri: "https://someurl.com", match: ["*.yml"]}]' }
+        let(:schema_content) { double(body: 'invalid json!') }
+
+        it 'returns schema successfully' do
+          expect(Gitlab::HTTP).to receive(:get).with("https://someurl.com")
+          expect(subject[:status]).to eq(:error)
+          expect(subject[:message]).to include('unexpected character () at line 1, column 1')
+        end
+      end
+
+      context 'schemas are defined and but no matching schema found' do
+        let(:config_content) { 'schemas: [{uri: "https://someurl.com", match: ["*.json"]}]' }
+
+        it 'returns empty schema object' do
+          expect(Gitlab::HTTP).not_to receive(:get).with("https://someurl.com")
+          expect(subject[:schema]).to eq({})
+        end
+      end
+
+      context 'nested schema filename with "**" in match uri' do
+        let(:config_content) { 'schemas: [{uri: "https://someurl.com", match: ["data/somepath/**/*.yml"]}]' }
+        let(:filename) { 'data/somepath/unreleased/changelog/path/changelog.yml' }
+
+        it 'returns schema successfully' do
+          expect(Gitlab::HTTP).to receive(:get).with("https://someurl.com")
+          expect(subject[:schema]['title']).to eq "Sample schema"
+        end
+      end
+    end
+  end
+end
diff --git a/lib/gitlab/web_ide/config.rb b/lib/gitlab/web_ide/config.rb
index 3b1fa162b533ace9ce67429b40fcfebea7d0ba14..b2ab5c0b6e39d1e9546b9452a12e3de29515877a 100644
--- a/lib/gitlab/web_ide/config.rb
+++ b/lib/gitlab/web_ide/config.rb
@@ -34,6 +34,10 @@ def terminal_value
         @global.terminal_value
       end
 
+      def schemas_value
+        @global.schemas_value
+      end
+
       private
 
       def build_config(config, opts = {})
diff --git a/lib/gitlab/web_ide/config/entry/global.rb b/lib/gitlab/web_ide/config/entry/global.rb
index 50c3f2d294f40731c4b44dbe7d024bc0baa93488..2c67c7d02d48b1625e82c00ad6fa4c91a46d5a44 100644
--- a/lib/gitlab/web_ide/config/entry/global.rb
+++ b/lib/gitlab/web_ide/config/entry/global.rb
@@ -12,18 +12,22 @@ class Global < ::Gitlab::Config::Entry::Node
           include ::Gitlab::Config::Entry::Configurable
           include ::Gitlab::Config::Entry::Attributable
 
-          ALLOWED_KEYS = %i[terminal].freeze
+          def self.allowed_keys
+            %i[terminal].freeze
+          end
 
           validations do
-            validates :config, allowed_keys: ALLOWED_KEYS
+            validates :config, allowed_keys: Global.allowed_keys
           end
 
+          attributes allowed_keys
+
           entry :terminal, Entry::Terminal,
             description: 'Configuration of the webide terminal.'
-
-          attributes :terminal
         end
       end
     end
   end
 end
+
+::Gitlab::WebIde::Config::Entry::Global.prepend_if_ee('EE::Gitlab::WebIde::Config::Entry::Global')
diff --git a/spec/controllers/projects/web_ide_schemas_controller_spec.rb b/spec/controllers/projects/web_ide_schemas_controller_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..fbec941aecc458626ec08bd027e098858ad3c6c1
--- /dev/null
+++ b/spec/controllers/projects/web_ide_schemas_controller_spec.rb
@@ -0,0 +1,66 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Projects::WebIdeSchemasController do
+  let_it_be(:developer) { create(:user) }
+  let_it_be(:project) { create(:project, :private, :repository, namespace: developer.namespace) }
+
+  before do
+    project.add_developer(developer)
+
+    sign_in(user)
+  end
+
+  describe 'GET show' do
+    let(:user) { developer }
+    let(:branch) { 'master' }
+
+    subject do
+      get :show, params: {
+        namespace_id: project.namespace.to_param,
+        project_id: project,
+        branch: branch,
+        filename: 'package.json'
+      }
+    end
+
+    before do
+      allow_next_instance_of(::Ide::SchemasConfigService) do |instance|
+        allow(instance).to receive(:execute).and_return(result)
+      end
+    end
+
+    context 'when branch is invalid' do
+      let(:branch) { 'non-existent' }
+
+      it 'returns 422' do
+        subject
+
+        expect(response).to have_gitlab_http_status(:unprocessable_entity)
+      end
+    end
+
+    context 'when a valid schema exists' do
+      let(:result) { { status: :success, schema: { schema: 'Sample Schema' } } }
+
+      it 'returns the schema' do
+        subject
+
+        expect(response).to have_gitlab_http_status(:ok)
+        expect(response.body).to eq('{"schema":"Sample Schema"}')
+      end
+    end
+
+    context 'when an error occurs parsing the schema' do
+      let(:result) { { status: :error, message: 'Some error occured' } }
+
+      it 'returns 422 with the error' do
+        subject
+
+        expect(response).to have_gitlab_http_status(:unprocessable_entity)
+        expect(response.body).to eq('{"status":"error","message":"Some error occured"}')
+      end
+    end
+  end
+end
diff --git a/spec/controllers/projects/web_ide_terminals_controller_spec.rb b/spec/controllers/projects/web_ide_terminals_controller_spec.rb
index 2ae5899c25822eef9fd2e8b0a08d6dad1323739f..3eb3d5da351a5f7ecfa7899036557fdf99869da8 100644
--- a/spec/controllers/projects/web_ide_terminals_controller_spec.rb
+++ b/spec/controllers/projects/web_ide_terminals_controller_spec.rb
@@ -113,7 +113,7 @@
     let(:result) { { status: :success } }
 
     before do
-      allow_next_instance_of(::Ci::WebIdeConfigService) do |instance|
+      allow_next_instance_of(::Ide::TerminalConfigService) do |instance|
         allow(instance).to receive(:execute).and_return(result)
       end
 
diff --git a/spec/lib/gitlab/web_ide/config/entry/global_spec.rb b/spec/lib/gitlab/web_ide/config/entry/global_spec.rb
index 3a50667163ba5b60239d9683f112e8deca05ab3e..3e29bf89785c5e6c7632168c6d63f47be5665ec7 100644
--- a/spec/lib/gitlab/web_ide/config/entry/global_spec.rb
+++ b/spec/lib/gitlab/web_ide/config/entry/global_spec.rb
@@ -12,8 +12,7 @@
 
     context 'when filtering all the entry/node names' do
       it 'contains the expected node names' do
-        expect(described_class.nodes.keys)
-          .to match_array(%i[terminal])
+        expect(described_class.nodes.keys).to match_array(described_class.allowed_keys)
       end
     end
   end
@@ -34,7 +33,7 @@
         end
 
         it 'creates node object for each entry' do
-          expect(global.descendants.count).to eq 1
+          expect(global.descendants.count).to eq described_class.allowed_keys.length
         end
 
         it 'creates node object using valid class' do
diff --git a/spec/services/ci/web_ide_config_service_spec.rb b/spec/services/ide/base_config_service_spec.rb
similarity index 53%
rename from spec/services/ci/web_ide_config_service_spec.rb
rename to spec/services/ide/base_config_service_spec.rb
index 437b468cec8f1dce3c910dc6cce4aa3fe20fad1b..debdc6e58096a66020dfc951f8079a9dda21729c 100644
--- a/spec/services/ci/web_ide_config_service_spec.rb
+++ b/spec/services/ide/base_config_service_spec.rb
@@ -2,7 +2,7 @@
 
 require 'spec_helper'
 
-RSpec.describe Ci::WebIdeConfigService do
+RSpec.describe Ide::BaseConfigService do
   let_it_be(:project) { create(:project, :repository) }
   let_it_be(:user) { create(:user) }
   let(:sha) { 'sha' }
@@ -47,44 +47,6 @@
               message: "Invalid configuration format")
           end
         end
-
-        context 'content is valid, but terminal not defined' do
-          let(:config_content) { '{}' }
-
-          it 'returns success' do
-            is_expected.to include(
-              status: :success,
-              terminal: nil)
-          end
-        end
-
-        context 'content is valid, with enabled terminal' do
-          let(:config_content) { 'terminal: {}' }
-
-          it 'returns success' do
-            is_expected.to include(
-              status: :success,
-              terminal: {
-                tag_list: [],
-                yaml_variables: [],
-                options: { script: ["sleep 60"] }
-              })
-          end
-        end
-
-        context 'content is valid, with custom terminal' do
-          let(:config_content) { 'terminal: { before_script: [ls] }' }
-
-          it 'returns success' do
-            is_expected.to include(
-              status: :success,
-              terminal: {
-                tag_list: [],
-                yaml_variables: [],
-                options: { before_script: ["ls"], script: ["sleep 60"] }
-              })
-          end
-        end
       end
     end
   end
diff --git a/spec/services/ide/schemas_config_service_spec.rb b/spec/services/ide/schemas_config_service_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..19e5ca9e87dca755a799916b3ec693bc749470dd
--- /dev/null
+++ b/spec/services/ide/schemas_config_service_spec.rb
@@ -0,0 +1,53 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Ide::SchemasConfigService do
+  let_it_be(:project) { create(:project, :repository) }
+  let_it_be(:user) { create(:user) }
+  let(:filename) { 'sample.yml' }
+  let(:schema_content) { double(body: '{"title":"Sample schema"}') }
+
+  describe '#execute' do
+    before do
+      project.add_developer(user)
+
+      allow(Gitlab::HTTP).to receive(:get).with(anything) do
+        schema_content
+      end
+    end
+
+    subject { described_class.new(project, user, filename: filename).execute }
+
+    context 'feature flag schema_linting is enabled', unless: Gitlab.ee? do
+      before do
+        stub_feature_flags(schema_linting: true)
+      end
+
+      context 'when no predefined schema exists for the given filename' do
+        it 'returns an empty object' do
+          is_expected.to include(
+            status: :success,
+            schema: {})
+        end
+      end
+
+      context 'when a predefined schema exists for the given filename' do
+        let(:filename) { '.gitlab-ci.yml' }
+
+        it 'uses predefined schema matches' do
+          expect(Gitlab::HTTP).to receive(:get).with('https://json.schemastore.org/gitlab-ci')
+          expect(subject[:schema]['title']).to eq "Sample schema"
+        end
+      end
+    end
+
+    context 'feature flag schema_linting is disabled', unless: Gitlab.ee? do
+      it 'returns an empty object' do
+        is_expected.to include(
+          status: :success,
+          schema: {})
+      end
+    end
+  end
+end
diff --git a/spec/services/ide/terminal_config_service_spec.rb b/spec/services/ide/terminal_config_service_spec.rb
new file mode 100644
index 0000000000000000000000000000000000000000..d6c4f7a2a69b47c2cbee62a4438354b297426611
--- /dev/null
+++ b/spec/services/ide/terminal_config_service_spec.rb
@@ -0,0 +1,69 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Ide::TerminalConfigService do
+  let_it_be(:project) { create(:project, :repository) }
+  let_it_be(:user) { create(:user) }
+  let(:sha) { 'sha' }
+
+  describe '#execute' do
+    subject { described_class.new(project, user, sha: sha).execute }
+
+    before do
+      project.add_developer(user)
+
+      allow(project.repository).to receive(:blob_data_at).with('sha', anything) do
+        config_content
+      end
+    end
+
+    context 'content is not valid' do
+      let(:config_content) { 'invalid content' }
+
+      it 'returns an error' do
+        is_expected.to include(
+          status: :error,
+          message: "Invalid configuration format")
+      end
+    end
+
+    context 'terminal not defined' do
+      let(:config_content) { '{}' }
+
+      it 'returns success' do
+        is_expected.to include(
+          status: :success,
+          terminal: nil)
+      end
+    end
+
+    context 'terminal enabled' do
+      let(:config_content) { 'terminal: {}' }
+
+      it 'returns success' do
+        is_expected.to include(
+          status: :success,
+          terminal: {
+            tag_list: [],
+            yaml_variables: [],
+            options: { script: ["sleep 60"] }
+          })
+      end
+    end
+
+    context 'custom terminal enabled' do
+      let(:config_content) { 'terminal: { before_script: [ls] }' }
+
+      it 'returns success' do
+        is_expected.to include(
+          status: :success,
+          terminal: {
+            tag_list: [],
+            yaml_variables: [],
+            options: { before_script: ["ls"], script: ["sleep 60"] }
+          })
+      end
+    end
+  end
+end