From b5bb479e5e2ab1f9756767c53492a4e315e3641b Mon Sep 17 00:00:00 2001
From: Peter Hegman <phegman@gitlab.com>
Date: Wed, 31 Jan 2024 12:25:23 +0000
Subject: [PATCH] Setup new group from organization route, controller, and Vue
 app

Allows user to create a group in an organization from the UI
---
 .../groups/new/components/app.vue             | 52 +++++++++++++++
 .../organizations/groups/new/index.js         | 45 +++++++++++++
 .../pages/organizations/groups/new/index.js   |  3 +
 .../organizations/application_controller.rb   |  4 ++
 .../organizations/groups_controller.rb        | 11 ++++
 .../organizations/organization_helper.rb      | 17 +++++
 app/views/layouts/organization.html.haml      |  2 +-
 app/views/organizations/groups/new.html.haml  |  4 ++
 config/routes/organizations.rb                |  2 +
 .../organizations/menus/manage_menu.rb        |  7 ++-
 .../groups/new/components/app_spec.js         | 47 ++++++++++++++
 .../organizations/organization_helper_spec.rb | 63 ++++++++++++++++++-
 .../organizations/groups_controller_spec.rb   | 50 +++++++++++++++
 .../groups_controller_routing_spec.rb         | 12 ++++
 .../layouts/organization.html.haml_spec.rb    | 43 ++-----------
 15 files changed, 319 insertions(+), 43 deletions(-)
 create mode 100644 app/assets/javascripts/organizations/groups/new/components/app.vue
 create mode 100644 app/assets/javascripts/organizations/groups/new/index.js
 create mode 100644 app/assets/javascripts/pages/organizations/groups/new/index.js
 create mode 100644 app/controllers/organizations/groups_controller.rb
 create mode 100644 app/views/organizations/groups/new.html.haml
 create mode 100644 spec/frontend/organizations/groups/new/components/app_spec.js
 create mode 100644 spec/requests/organizations/groups_controller_spec.rb
 create mode 100644 spec/routing/organizations/groups_controller_routing_spec.rb

diff --git a/app/assets/javascripts/organizations/groups/new/components/app.vue b/app/assets/javascripts/organizations/groups/new/components/app.vue
new file mode 100644
index 0000000000000..a8ff505f4f249
--- /dev/null
+++ b/app/assets/javascripts/organizations/groups/new/components/app.vue
@@ -0,0 +1,52 @@
+<script>
+import { GlSprintf, GlLink } from '@gitlab/ui';
+import { s__, __ } from '~/locale';
+import { helpPagePath } from '~/helpers/help_page_helper';
+
+export default {
+  name: 'OrganizationGroupsNewApp',
+  i18n: {
+    pageTitle: __('New group'),
+    description1: s__(
+      'GroupsNew|%{linkStart}Groups%{linkEnd} allow you to manage and collaborate across multiple projects. Members of a group have access to all of its projects.',
+    ),
+    description2: s__(
+      'GroupsNew|Groups can also be nested by creating %{linkStart}subgroups%{linkEnd}.',
+    ),
+  },
+  groupsHelpPagePath: helpPagePath('user/group/index'),
+  subgroupsHelpPagePath: helpPagePath('user/group/subgroups/index'),
+  components: {
+    GlLink,
+    GlSprintf,
+  },
+  inject: [
+    'organizationId',
+    'basePath',
+    'groupsOrganizationPath',
+    'mattermostEnabled',
+    'availableVisibilityLevels',
+    'restrictedVisibilityLevels',
+  ],
+};
+</script>
+
+<template>
+  <div class="gl-py-6">
+    <h1 class="gl-mt-0 gl-font-size-h-display">{{ $options.i18n.pageTitle }}</h1>
+    <p>
+      <gl-sprintf :message="$options.i18n.description1">
+        <template #link="{ content }">
+          <gl-link :href="$options.groupsHelpPagePath" target="_blank">{{ content }}</gl-link>
+        </template>
+      </gl-sprintf>
+    </p>
+    <p>
+      <gl-sprintf :message="$options.i18n.description2">
+        <template #link="{ content }">
+          <gl-link :href="$options.subgroupsHelpPagePath" target="_blank">{{ content }}</gl-link>
+        </template>
+      </gl-sprintf>
+    </p>
+  </div>
+</template>
diff --git a/app/assets/javascripts/organizations/groups/new/index.js b/app/assets/javascripts/organizations/groups/new/index.js
new file mode 100644
index 0000000000000..730b7293e660d
--- /dev/null
+++ b/app/assets/javascripts/organizations/groups/new/index.js
@@ -0,0 +1,45 @@
+import Vue from 'vue';
+import VueApollo from 'vue-apollo';
+
+import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
+import createDefaultClient from '~/lib/graphql';
+import App from './components/app.vue';
+
+export const initOrganizationsGroupsNew = () => {
+  const el = document.getElementById('js-organizations-groups-new');
+
+  if (!el) return false;
+
+  const {
+    dataset: { appData },
+  } = el;
+  const {
+    organizationId,
+    basePath,
+    groupsOrganizationPath,
+    mattermostEnabled,
+    availableVisibilityLevels,
+    restrictedVisibilityLevels,
+  } = convertObjectPropsToCamelCase(JSON.parse(appData));
+
+  const apolloProvider = new VueApollo({
+    defaultClient: createDefaultClient(),
+  });
+
+  return new Vue({
+    el,
+    name: 'OrganizationGroupsNewRoot',
+    apolloProvider,
+    provide: {
+      organizationId,
+      basePath,
+      groupsOrganizationPath,
+      mattermostEnabled,
+      availableVisibilityLevels,
+      restrictedVisibilityLevels,
+    },
+    render(createElement) {
+      return createElement(App);
+    },
+  });
+};
diff --git a/app/assets/javascripts/pages/organizations/groups/new/index.js b/app/assets/javascripts/pages/organizations/groups/new/index.js
new file mode 100644
index 0000000000000..6feb5a7eeb949
--- /dev/null
+++ b/app/assets/javascripts/pages/organizations/groups/new/index.js
@@ -0,0 +1,3 @@
+import { initOrganizationsGroupsNew } from '~/organizations/groups/new';
+
+initOrganizationsGroupsNew();
diff --git a/app/controllers/organizations/application_controller.rb b/app/controllers/organizations/application_controller.rb
index 9cc33ec04471a..b4a050835c0d5 100644
--- a/app/controllers/organizations/application_controller.rb
+++ b/app/controllers/organizations/application_controller.rb
@@ -35,5 +35,9 @@ def authorize_read_organization_user!
     def authorize_admin_organization!
       access_denied! unless can?(current_user, :admin_organization, organization)
     end
+
+    def authorize_create_group!
+      access_denied! unless can?(current_user, :create_group, organization)
+    end
   end
 end
diff --git a/app/controllers/organizations/groups_controller.rb b/app/controllers/organizations/groups_controller.rb
new file mode 100644
index 0000000000000..104b8a653f853
--- /dev/null
+++ b/app/controllers/organizations/groups_controller.rb
@@ -0,0 +1,11 @@
+# frozen_string_literal: true
+
+module Organizations
+  class GroupsController < ApplicationController
+    feature_category :cell
+
+    def new
+      authorize_create_group!
+    end
+  end
+end
diff --git a/app/helpers/organizations/organization_helper.rb b/app/helpers/organizations/organization_helper.rb
index 85f31e4cb9918..e18cd409d73dc 100644
--- a/app/helpers/organizations/organization_helper.rb
+++ b/app/helpers/organizations/organization_helper.rb
@@ -2,6 +2,12 @@
 
 module Organizations
   module OrganizationHelper
+    def organization_layout_nav
+      return 'organization' unless current_controller?('organizations')
+
+      current_action?(:index, :new) ? "your_work" : "organization"
+    end
+
     def organization_show_app_data(organization)
       {
         organization: organization.slice(:id, :name, :description_html)
@@ -52,6 +58,17 @@ def home_organization_setting_app_data
       }.to_json
     end
 
+    def organization_groups_new_app_data(organization)
+      {
+        organization_id: organization.id,
+        base_path: root_url,
+        groups_organization_path: groups_and_projects_organization_path(organization, { display: 'groups' }),
+        mattermost_enabled: Gitlab.config.mattermost.enabled,
+        available_visibility_levels: available_visibility_levels(Group),
+        restricted_visibility_levels: restricted_visibility_levels
+      }.to_json
+    end
+
     private
 
     def shared_groups_and_projects_app_data(organization)
diff --git a/app/views/layouts/organization.html.haml b/app/views/layouts/organization.html.haml
index a4485bb791e47..41f88c2379deb 100644
--- a/app/views/layouts/organization.html.haml
+++ b/app/views/layouts/organization.html.haml
@@ -1,5 +1,5 @@
 - page_title @organization.name if @organization
 - header_title @organization.name, organization_path(@organization) if @organization
-- nav(%w[index new].include?(params[:action]) ? "your_work" : "organization")
+- nav(organization_layout_nav)
 
 = render template: "layouts/application"
diff --git a/app/views/organizations/groups/new.html.haml b/app/views/organizations/groups/new.html.haml
new file mode 100644
index 0000000000000..f02e9f203d625
--- /dev/null
+++ b/app/views/organizations/groups/new.html.haml
@@ -0,0 +1,4 @@
+- page_title _('New group')
+- add_to_breadcrumbs _('Groups and projects'), groups_and_projects_organization_path(@organization)
+
+#js-organizations-groups-new{ data: { app_data: organization_groups_new_app_data(@organization) } }
diff --git a/config/routes/organizations.rb b/config/routes/organizations.rb
index dbc9f2ce2260a..00a00b25e9fe1 100644
--- a/config/routes/organizations.rb
+++ b/config/routes/organizations.rb
@@ -17,5 +17,7 @@
     resource :settings, only: [], as: :settings_organization do
       get :general
     end
+
+    resource :groups, only: [:new], as: :groups_organization
   end
 end
diff --git a/lib/sidebars/organizations/menus/manage_menu.rb b/lib/sidebars/organizations/menus/manage_menu.rb
index cdbc5d16f1bc7..1cea6ebe89706 100644
--- a/lib/sidebars/organizations/menus/manage_menu.rb
+++ b/lib/sidebars/organizations/menus/manage_menu.rb
@@ -33,7 +33,12 @@ def groups_and_projects_menu_item
               title: _('Groups and projects'),
               link: groups_and_projects_organization_path(context.container),
               super_sidebar_parent: ::Sidebars::Organizations::Menus::ManageMenu,
-              active_routes: { path: 'organizations/organizations#groups_and_projects' },
+              active_routes: {
+                path: %w[
+                  organizations/organizations#groups_and_projects
+                  organizations/groups#new
+                ]
+              },
               item_id: :organization_groups_and_projects
             )
           )
diff --git a/spec/frontend/organizations/groups/new/components/app_spec.js b/spec/frontend/organizations/groups/new/components/app_spec.js
new file mode 100644
index 0000000000000..f1eeb264d3189
--- /dev/null
+++ b/spec/frontend/organizations/groups/new/components/app_spec.js
@@ -0,0 +1,47 @@
+import { GlSprintf, GlLink } from '@gitlab/ui';
+import { shallowMountExtended } from 'helpers/vue_test_utils_helper';
+import App from '~/organizations/groups/new/components/app.vue';
+import { helpPagePath } from '~/helpers/help_page_helper';
+
+describe('OrganizationGroupsNewApp', () => {
+  let wrapper;
+
+  const defaultProvide = {
+    organizationId: 1,
+    basePath: 'https://gitlab.com',
+    groupsOrganizationPath: '/-/organizations/carrot/groups_and_projects?display=groups',
+    mattermostEnabled: false,
+    availableVisibilityLevels: [0, 10, 20],
+    restrictedVisibilityLevels: [],
+  };
+
+  const createComponent = () => {
+    wrapper = shallowMountExtended(App, {
+      provide: defaultProvide,
+      stubs: {
+        GlSprintf,
+        GlLink,
+      },
+    });
+  };
+
+  const findAllParagraphs = () => wrapper.findAll('p');
+  const findAllLinks = () => wrapper.findAllComponents(GlLink);
+
+  it('renders page title and description', () => {
+    createComponent();
+
+    expect(wrapper.findByRole('heading', { name: 'New group' }).exists()).toBe(true);
+
+    expect(findAllParagraphs().at(0).text()).toMatchInterpolatedText(
+      'Groups allow you to manage and collaborate across multiple projects. Members of a group have access to all of its projects.',
+    );
+    expect(findAllLinks().at(0).attributes('href')).toBe(helpPagePath('user/group/index'));
+    expect(findAllParagraphs().at(1).text()).toContain(
+      'Groups can also be nested by creating subgroups.',
+    );
+    expect(findAllLinks().at(1).attributes('href')).toBe(
+      helpPagePath('user/group/subgroups/index'),
+    );
+  });
+});
diff --git a/spec/helpers/organizations/organization_helper_spec.rb b/spec/helpers/organizations/organization_helper_spec.rb
index 0535a860b9426..1ec95e4503ad1 100644
--- a/spec/helpers/organizations/organization_helper_spec.rb
+++ b/spec/helpers/organizations/organization_helper_spec.rb
@@ -14,6 +14,7 @@
   let_it_be(:groups_empty_state_svg_path) { 'illustrations/empty-state/empty-groups-md.svg' }
   let_it_be(:projects_empty_state_svg_path) { 'illustrations/empty-state/empty-projects-md.svg' }
   let_it_be(:preview_markdown_organizations_path) { '/-/organizations/preview_markdown' }
+  let_it_be(:groups_and_projects_organization_path) { '/-/organizations/default/groups_and_projects' }
 
   before do
     allow(organization).to receive(:to_global_id).and_return(organization_gid)
@@ -28,11 +29,43 @@
     allow(helper).to receive(:preview_markdown_organizations_path).and_return(preview_markdown_organizations_path)
   end
 
+  describe '#organization_layout_nav' do
+    context 'when current controller is not organizations' do
+      it 'returns organization' do
+        allow(helper).to receive(:current_controller?).with('organizations').and_return(false)
+
+        expect(helper.organization_layout_nav).to eq('organization')
+      end
+    end
+
+    context 'when current controller is organizations' do
+      before do
+        allow(helper).to receive(:current_controller?).with('organizations').and_return(true)
+      end
+
+      context 'when current action is index or new' do
+        it 'returns your_work' do
+          allow(helper).to receive(:current_action?).with(:index, :new).and_return(true)
+
+          expect(helper.organization_layout_nav).to eq('your_work')
+        end
+      end
+
+      context 'when current action is not index or new' do
+        it 'returns organization' do
+          allow(helper).to receive(:current_action?).with(:index, :new).and_return(false)
+
+          expect(helper.organization_layout_nav).to eq('organization')
+        end
+      end
+    end
+  end
+
   describe '#organization_show_app_data' do
     before do
       allow(helper).to receive(:groups_and_projects_organization_path)
         .with(organization)
-        .and_return('/-/organizations/default/groups_and_projects')
+        .and_return(groups_and_projects_organization_path)
     end
 
     it 'returns expected json' do
@@ -50,7 +83,7 @@
             'description_html' => organization.description_html,
             'avatar_url' => 'avatar.jpg'
           },
-          'groups_and_projects_organization_path' => '/-/organizations/default/groups_and_projects',
+          'groups_and_projects_organization_path' => groups_and_projects_organization_path,
           'new_group_path' => new_group_path,
           'new_project_path' => new_project_path,
           'groups_empty_state_svg_path' => groups_empty_state_svg_path,
@@ -151,4 +184,30 @@
       )
     end
   end
+
+  describe '#organization_groups_new_app_data' do
+    before do
+      allow(helper).to receive(:groups_and_projects_organization_path)
+        .with(organization, { display: 'groups' })
+        .and_return(groups_and_projects_organization_path)
+      allow(helper).to receive(:restricted_visibility_levels).and_return([])
+    end
+
+    it 'returns expected json' do
+      expect(Gitlab::Json.parse(helper.organization_groups_new_app_data(organization))).to eq(
+        {
+          'organization_id' => organization.id,
+          'base_path' => root_url,
+          'groups_organization_path' => groups_and_projects_organization_path,
+          'mattermost_enabled' => false,
+          'available_visibility_levels' => [
+            Gitlab::VisibilityLevel::PRIVATE,
+            Gitlab::VisibilityLevel::INTERNAL,
+            Gitlab::VisibilityLevel::PUBLIC
+          ],
+          'restricted_visibility_levels' => []
+        }
+      )
+    end
+  end
 end
diff --git a/spec/requests/organizations/groups_controller_spec.rb b/spec/requests/organizations/groups_controller_spec.rb
new file mode 100644
index 0000000000000..7c5f019a41c74
--- /dev/null
+++ b/spec/requests/organizations/groups_controller_spec.rb
@@ -0,0 +1,50 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Organizations::GroupsController, feature_category: :cell do
+  let_it_be(:organization) { create(:organization) }
+
+  describe 'GET #new' do
+    subject(:gitlab_request) { get new_groups_organization_path(organization) }
+
+    context 'when the user is not signed in' do
+      it_behaves_like 'organization - redirects to sign in page'
+
+      context 'when `ui_for_organizations` feature flag is disabled' do
+        before do
+          stub_feature_flags(ui_for_organizations: false)
+        end
+
+        it_behaves_like 'organization - redirects to sign in page'
+      end
+    end
+
+    context 'when the user is signed in' do
+      let_it_be(:user) { create(:user) }
+
+      before do
+        sign_in(user)
+      end
+
+      context 'with no association to an organization' do
+        it_behaves_like 'organization - not found response'
+        it_behaves_like 'organization - action disabled by `ui_for_organizations` feature flag'
+      end
+
+      context 'as as admin', :enable_admin_mode do
+        let_it_be(:user) { create(:admin) }
+
+        it_behaves_like 'organization - successful response'
+        it_behaves_like 'organization - action disabled by `ui_for_organizations` feature flag'
+      end
+
+      context 'as an organization user' do
+        let_it_be(:organization_user) { create(:organization_user, organization: organization, user: user) }
+
+        it_behaves_like 'organization - successful response'
+        it_behaves_like 'organization - action disabled by `ui_for_organizations` feature flag'
+      end
+    end
+  end
+end
diff --git a/spec/routing/organizations/groups_controller_routing_spec.rb b/spec/routing/organizations/groups_controller_routing_spec.rb
new file mode 100644
index 0000000000000..34388003a652b
--- /dev/null
+++ b/spec/routing/organizations/groups_controller_routing_spec.rb
@@ -0,0 +1,12 @@
+# frozen_string_literal: true
+
+require 'spec_helper'
+
+RSpec.describe Organizations::GroupsController, :routing, feature_category: :cell do
+  let_it_be(:organization) { build(:organization) }
+
+  it 'routes to groups#new' do
+    expect(get("/-/organizations/#{organization.path}/groups/new"))
+      .to route_to('organizations/groups#new', organization_path: organization.path)
+  end
+end
diff --git a/spec/views/layouts/organization.html.haml_spec.rb b/spec/views/layouts/organization.html.haml_spec.rb
index 72a4abd288ab5..53818a161d68d 100644
--- a/spec/views/layouts/organization.html.haml_spec.rb
+++ b/spec/views/layouts/organization.html.haml_spec.rb
@@ -12,48 +12,13 @@
     allow(view).to receive(:users_path).and_return('/root')
   end
 
-  subject do
-    render
-
-    rendered
-  end
-
   describe 'navigation' do
-    context 'when action is #index' do
-      before do
-        allow(view).to receive(:params).and_return({ action: 'index' })
-      end
-
-      it 'renders your_work navigation' do
-        subject
-
-        expect(view.instance_variable_get(:@nav)).to eq('your_work')
-      end
-    end
-
-    context 'when action is #new' do
-      before do
-        allow(view).to receive(:params).and_return({ action: 'new' })
-      end
-
-      it 'renders your_work navigation' do
-        subject
-
-        expect(view.instance_variable_get(:@nav)).to eq('your_work')
-      end
-    end
-
-    context 'when action is #show' do
-      before do
-        allow(view).to receive(:params).and_return({ action: 'show' })
-        view.instance_variable_set(:@organization, organization)
-      end
+    it 'calls organization_layout_nav and sets @nav instance variable' do
+      expect(view).to receive(:organization_layout_nav).and_return('your_work')
 
-      it 'renders organization navigation' do
-        subject
+      render
 
-        expect(view.instance_variable_get(:@nav)).to eq('organization')
-      end
+      expect(view.instance_variable_get(:@nav)).to eq('your_work')
     end
   end
 end
-- 
GitLab